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

前言

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

示例代码

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

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

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

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

  private static Func 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>(memberInitExpression, parameterExpression);
   return lambda.Compile();
  }

  /// 
  /// 根据转换源和目标获取表达式树
  /// 
  /// 表达式参数p
  /// 转换源类型
  /// 转换目标类型
  /// 
  private static MemberInitExpression GetExpression(Expression parameterExpression, Type sourceType, Type targetType)
  {
   var memberBindings = new List();
   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() != 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; }
 
 /// 
 /// 标注为notmapped特性时,不转换赋值
 /// 
 [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; }
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 = "李四"
 }
};
B b = Mapper.Map(a);//得到转换结果

4.性能测试

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

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

总结

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

你可能感兴趣的:(.NET core高性能对象转换示例代码)