功能一:
背景:
编程中经常有这个需求,要在两个非常类似的实体类中,拷贝同名属性的值,(例如在WCF实体和EF实体中的拷贝...)
以往一般有两个方案
1.硬编码:执行效率很高,不过要写很多重复的代码,
2.反射: 灵活,不过效率非常低
这里提供一个灵活性不比反射差的解决方案 https://github.com/xwj90/Clover.Copyer
使用代码非常简单,如下所示, 只有一句话
//范例一 在两个对象直接拷贝属性
ClassA target = new ClassA();
ClassB source = new ClassB() { A = "AA", B = 2, C = DateTime.Now };
//复制代码
CopyHelper<ClassB, ClassA>.Copy(source, target);//这一句是调用方法
Console.WriteLine(target.A);
Console.WriteLine(target.B);
Console.WriteLine(target.C);
两个类型的定义如下:
public sealed class ClassA
{
public string A { get; set; }
public int B { get; set; }
public string Title { get; set; }
public int Id { get; set; }
public DateTime C { get; set; }
}
public sealed class ClassB
{
public string A { get; set; }
public int B { get; set; }
public DateTime C { get; set; }
}
备注:支持所有类型的公开实例属性
不支持索引器
功能二:
从数据库提供的IDataReader中读取出对应的值,并转换成相应类型赋值给实体类,
省掉了中间的Dataset的过程,也节省了很多类型转换和硬编码的工作
先看看代码:
//范例二 从IDataReader中读取属性
string connString = "Data Source=.;Initial Catalog=Northwind;Persist Security Info=True;";
for (int i = 0; i < 1; i++)
{
using (SqlConnection conn = new SqlConnection(connString))
{
SqlCommand command = new SqlCommand();
command.Connection = conn;
command.CommandText = "select top 1000 * from cLog";
conn.Open();
IDataReader reader = command.ExecuteReader();
var list = CopyHelper<TestClassA>.Copy(reader);//这一句是调用方法
foreach (var item in list)
{
Console.Write(item.LogId + "---");
Console.Write(item.LogTime + "---");
Console.Write(item.AppName + "---");
Console.Write(item.Server + "---");
Console.Write(item.IPAddress + " ");
Console.WriteLine();
}
}
}
备注:支持所有基本类型,每次返回的列的顺序要一样, 不要一会儿select a,b from table 然后一会儿 select b,a from table (自动探测要消耗很多性能,以后再实现)
实现原理:
第一次的时候使用反射分析类型, 然后使用ILEmit 生成动态方法
将动态方法缓存下来,第二次的时候就可以直接使用该方法了
性能:
基本上在属性比较多的类型中效率比例大约为
硬编码:动态方法:反射
1:5-10:100-1000 (实际测试的时候类中大约有 1-30个属性)
还请各位提点意见