.NET core高性能对象转换示例代码

 更新时间:2021年9月22日 10:02  点击:1765

前言

NET Core(开放源代码,跨平台,x-copy可部署等)有许多令人兴奋的方面,其中最值得称赞的就是其性能了。关于对象转换已经有不少轮子(AutoMapper,TinyMapper) .出于项目需要,手动造一个简单轮子。下面话不多说了,来一起看看详细的介绍吧。

示例代码

g>1.采用静态泛型类缓存,避免了拆箱装箱操作。

2.对于转换对象中有,字段名一样但是类型不一样的类时仍可以用

public static class Mapper<TSource, TTarget> where TSource : class where TTarget : class
 {
  public readonly static Func<TSource, TTarget> Map;

  static Mapper()
  {
   if (Map == null)
    Map = GetMap();
  }

  private static Func<TSource, TTarget> GetMap()
  {
   var sourceType = typeof(TSource);
   var targetType = typeof(TTarget);

   var parameterExpression = Expression.Parameter(sourceType, "p");
   var memberInitExpression = GetExpression(parameterExpression, sourceType, targetType);

   var lambda = Expression.Lambda<Func<TSource, TTarget>>(memberInitExpression, parameterExpression);
   return lambda.Compile();
  }

  /// <summary>
  /// 根据转换源和目标获取表达式树
  /// </summary>
  /// <param name="parameterExpression">表达式参数p</param>
  /// <param name="sourceType">转换源类型</param>
  /// <param name="targetType">转换目标类型</param>
  /// <returns></returns>
  private static MemberInitExpression GetExpression(Expression parameterExpression, Type sourceType, Type targetType)
  {
   var memberBindings = new List<MemberBinding>();
   foreach (var targetItem in targetType.GetProperties().Where(x => x.PropertyType.IsPublic && x.CanWrite))
   {
    var sourceItem = sourceType.GetProperty(targetItem.Name);

    //判断实体的读写权限
    if (sourceItem == null || !sourceItem.CanRead || sourceItem.PropertyType.IsNotPublic)
     continue;

    //标注NotMapped特性的属性忽略转换
    if (sourceItem.GetCustomAttribute<NotMappedAttribute>() != null)
     continue;

    var propertyExpression = Expression.Property(parameterExpression, sourceItem);

    //判断都是class 且类型不相同时
    if (targetItem.PropertyType.IsClass && sourceItem.PropertyType.IsClass && targetItem.PropertyType != sourceItem.PropertyType)
    {
     if (targetItem.PropertyType != targetType)//防止出现自己引用自己无限递归
     {
      var memberInit = GetExpression(propertyExpression, sourceItem.PropertyType, targetItem.PropertyType);
      memberBindings.Add(Expression.Bind(targetItem, memberInit));
      continue;
     }
    }

    if (targetItem.PropertyType != sourceItem.PropertyType)
     continue;

    memberBindings.Add(Expression.Bind(targetItem, propertyExpression));
   }
   return Expression.MemberInit(Expression.New(targetType), memberBindings);
  }
 }

3.调用方法如下

 (1)构造样例类

public class A
{
 public int Id { get; set; }
 public string Name { get; set; }
 public C User { get; set; }
 
 /// <summary>
 /// 标注为notmapped特性时,不转换赋值
 /// </summary>
 [System.ComponentModel.DataAnnotations.Schema.NotMapped]
 public D UserA { get; set; }
 
}
 
public class B
{
 public int Id { get; set; }
 public string Name { get; set; }
 public D User { get; set; }<br data-filtered="filtered">
 public D UserA { get; set; }
}
 
public class C
{
 public int Id { get; set; }
 public string Name { get; set; }
}
 
public class D
{
 public int Id { get; set; }
 public string Name { get; set; }
}

  (2) 调用

var a = new A
{
 Id = 1,
 Name = "张三",
 User = new C
 {
  Id = 1,
  Name = "李四"
 }
};<br>
B b = Mapper<A, B>.Map(a);//得到转换结果

4.性能测试

var length = 10000000;
   var listA = new List<A>();
   for (int i = 0; i < length; i++)
   {
    listA.Add(new A
    {
     Id = i,
     Name = "张三",
     User = new C
     {
      Id = i,
      Name = "李四"
     }
    });
   }

   var sw = Stopwatch.StartNew();
   for (int i = 0; i < length; i++)
   {
    var item = listA[i];
    var b = new B
    {
     Id = item.Id,
     Name = item.Name,
     User = new D
     {
      Id = i,
      Name = "李四",
     }
    };
   }
   sw.Stop();
   Console.WriteLine($"原生的时间:{sw.ElapsedMilliseconds}ms");

   //表达式
   Mapper<A, B>.Map(listA[0]);//预先编译缓存
   sw.Restart();
   for (int i = 0; i < length; i++)
   {
    Mapper<A, B>.Map(listA[i]);
   }
   sw.Stop();
   Console.WriteLine($"表达式的时间:{sw.ElapsedMilliseconds}ms");

   //AutoMapper
   AutoMapper.Mapper.Initialize(cfg => cfg.CreateMap<A, B>());
   sw.Restart();
   for (int i = 0; i < length; i++)
   {
    var b = AutoMapper.Mapper.Map<B>(listA[i]);
   }
   sw.Stop();
   Console.WriteLine($"AutoMapper时间:{sw.ElapsedMilliseconds}ms");

   //TinyMapper
   TinyMapper.Bind<A, B>();
   sw.Restart();
   for (int i = 0; i < length; i++)
   {
    var b = TinyMapper.Map<B>(listA[i]);
   }
   sw.Stop();
   Console.WriteLine($"TinyMapper时间:{sw.ElapsedMilliseconds}ms");

   Console.ReadLine();

5. 1000万数据不带子类集结果

6. 1000万数据带子类集结果 

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对猪先飞的支持。

[!--infotagslink--]

相关文章

  • .NET Core下使用Kafka的方法步骤

    这篇文章主要介绍了.NET Core下使用Kafka的方法步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-09-22
  • 详解.NET Core 3.0 里新的JSON API

    这篇文章主要介绍了详解.NET Core 3.0 里新的JSON API,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-09-22
  • ASP.NET Core根据环境变量支持多个 appsettings.json配置文件

    这篇文章主要介绍了ASP.NET Core根据环境变量支持多个 appsettings.json配置文件,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-09-22
  • 记一次EFCore类型转换错误及解决方案

    这篇文章主要介绍了记一次EFCore类型转换错误及解决方案,帮助大家更好的理解和学习使用asp.net core,感兴趣的朋友可以了解下...2021-09-22
  • 详解ASP.NET Core 中基于工厂的中间件激活的实现方法

    这篇文章主要介绍了ASP.NET Core 中基于工厂的中间件激活的实现方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-09-22
  • Underscore源码分析

    Underscore 是一个 JavaScript 工具库,它提供了一整套函数式编程的实用功能,但是没有扩展任何 JavaScript 内置对象。这篇文章主要介绍了underscore源码分析相关知识,感兴趣的朋友一起学习吧...2016-01-02
  • 详解.NET Core 使用HttpClient SSL请求出错的解决办法

    这篇文章主要介绍了.NET Core 使用HttpClient SSL请求出错的解决办法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2021-09-22
  • .net EF Core专题:EF Core 读取数据时发生了什么?

    这篇文章主要介绍了EF Core 读取数据的的相关知识,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下...2021-09-22
  • 详解ASP.NET Core Token认证

    这篇文章主要介绍了详解ASP.NET Core Token认证,小编觉得挺不错的,现在分享给大家,也给大家做个参考。...2021-09-22
  • .NET Core如何获取操作系统中的各种信息

    .net core是最近讨论频率很高的话题,下面这篇文章主要给大家介绍了关于利用.NET Core如何获取操作系统中各种信息的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面来一起看看吧...2021-09-22
  • asp.net core MVC之实现基于token的认证

    这篇文章主要介绍了asp.net core MVC之实现基于token的认证,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-05-07
  • ASP.NET Core如何注入多个服务实现类

    这篇文章主要介绍了ASP.NET Core如何注入多个服务实现类的相关资料,需要的朋友可以参考下面文章的具体内容...2021-09-22
  • ASP.NET core Web中使用appsettings.json配置文件的方法

    这篇文章主要给大家介绍了在ASP.NET core Web中使用appsettings.json配置文件的方法,文中给出了详细的示例代码,需要的朋友可以参考学习,下面来一起看看吧。...2021-09-22
  • ASP.NET Core使用JWT认证授权的方法

    这篇文章主要介绍了ASP.NET Core使用JWT认证授权的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-09-22
  • 详解EFCore中的导航属性

    这篇文章主要介绍了详解EFCore中的导航属性的相关资料,帮助大家更好的理解和学习使用ASP.NET CORE,感兴趣的朋友可以了解下...2021-09-22
  • ASP.NET Core MVC如何实现运行时动态定义Controller类型

    这篇文章主要介绍了ASP.NET Core MVC如何实现运行时动态定义Controller类型,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-09-22
  • dotnet core链接mongodb代码实例

    这篇文章主要介绍了dotnet core链接mongodb代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下...2020-06-25
  • 详解ASP.NET Core部署项目到Ubuntu Server

    这篇文章主要介绍了详解ASP.NET Core部署项目到Ubuntu Server ,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-09-22
  • ASP.NET Core使用SkiaSharp实现验证码的示例代码

    本篇文章主要介绍了ASP.NET Core使用SkiaSharp实现验证码的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2021-09-22
  • 利用.net core实现反向代理中间件的方法

    这篇文章主要给大家介绍了关于利用.net core实现反向代理中间件的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用.net core具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧...2021-09-22