前言
NET Core(开放源代码,跨平台,x-copy可部署等)有许多令人兴奋的方面,其中最值得称赞的就是其性能了。关于对象转换已经有不少轮子(AutoMapper,TinyMapper) .出于项目需要,手动造一个简单轮子。下面话不多说了,来一起看看详细的介绍吧。
示例代码
g>1.采用静态泛型类缓存,避免了拆箱装箱操作。
2.对于转换对象中有,字段名一样但是类型不一样的类时仍可以用
public static class Mapperwhere 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.性能测试
var length = 10000000;
var listA = new List();
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.Map(listA[0]);//预先编译缓存
sw.Restart();
for (int i = 0; i < length; i++)
{
Mapper.Map(listA[i]);
}
sw.Stop();
Console.WriteLine($"表达式的时间:{sw.ElapsedMilliseconds}ms");
//AutoMapper
AutoMapper.Mapper.Initialize(cfg => cfg.CreateMap());
sw.Restart();
for (int i = 0; i < length; i++)
{
var b = AutoMapper.Mapper.Map(listA[i]);
}
sw.Stop();
Console.WriteLine($"AutoMapper时间:{sw.ElapsedMilliseconds}ms");
//TinyMapper
TinyMapper.Bind();
sw.Restart();
for (int i = 0; i < length; i++)
{
var b = TinyMapper.Map(listA[i]);
}
sw.Stop();
Console.WriteLine($"TinyMapper时间:{sw.ElapsedMilliseconds}ms");
Console.ReadLine();
5. 1000万数据不带子类集结果
6. 1000万数据带子类集结果
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对考高分网的支持。



