C#中DataTable 转实体实例详解
因为Linq的查询功能很强大,所以从数据库中拿到的数据为了处理方便,我都会转换成实体集合List
开始用的是硬编码的方式,好理解,但通用性极低,下面是控件台中的代码:
usingSystem;
usingSystem.Collections.Generic;
usingSystem.Data;
usingSystem.Linq;
usingSystem.Text;
usingSystem.Threading.Tasks;
namespaceDemo1
{
classProgram
{
staticvoidMain(string[]args)
{
DataTabledt=Query();
Listusrs=newList(dt.Rows.Count);
//硬编码,效率比较高,但灵活性不够,如果实体改变了,都需要修改代码
foreach(DataRowdrindt.Rows)
{
Usrusr=newUsr{ID=dr.Field("ID"),Name=dr.Field("Name")};
usrs.Add(usr);
}
usrs.Clear();
}
///
///查询数据
///
///
privatestaticDataTableQuery()
{
DataTabledt=newDataTable();
dt.Columns.Add("ID",typeof(Int32));
dt.Columns.Add("Name",typeof(String));
for(inti=0;i<1000000;i++)
{
dt.Rows.Add(newObject[]{i,Guid.NewGuid().ToString()});
}
returndt;
}
}
classUsr
{
publicInt32?ID{get;set;}
publicStringName{get;set;}
}
}
后来用反射来做这,对实体的属性用反射去赋值,这样就可以对所有的实体通用,且增加属性后不用修改代码。
程序如下:
staticclassEntityConvert
{
///
///DataTable转为List
///
///
///
///
publicstaticListToList(thisDataTabledt)whereT:class,new()
{
Listcolletion=newList();
PropertyInfo[]pInfos=typeof(T).GetProperties();
foreach(DataRowdrindt.Rows)
{
Tt=newT();
foreach(PropertyInfopInfoinpInfos)
{
if(!pInfo.CanWrite)continue;
pInfo.SetValue(t,dr[pInfo.Name]);
}
colletion.Add(t);
}
returncolletion;
}
}
增加一个扩展方法,程序更加通用。但效率不怎么样,100万行数据【只有两列】,转换需要2秒
后来想到用委托去做委托原型如下
Funcfunc=dr=>newUsr{ID=dr.Field ("ID"),Name=dr.Field ("Name")};
代码如下:
staticvoidMain(string[]args)
{
DataTabledt=Query();
Funcfunc=dr=>newUsr{ID=dr.Field("ID"),Name=dr.Field("Name")};
Listusrs=newList(dt.Rows.Count);
Stopwatchsw=Stopwatch.StartNew();
foreach(DataRowdrindt.Rows)
{
Usrusr=func(dr);
usrs.Add(usr);
}
sw.Stop();
Console.WriteLine(sw.ElapsedMilliseconds);
usrs.Clear();
Console.ReadKey();
}
速度确实快了很多,我电脑测试了一下,需要0.4秒。但问题又来了,这个只能用于Usr这个类,得想办法把这个类抽象成泛型T,既有委托的高效,又有泛型的通用。
问题就在动态地产生上面的委托了,经过一下午的折腾终于折腾出来了动态产生委托的方法。主要用到了动态Lambda表达式
publicstaticclassEntityConverter
{
///
///DataTable生成实体
///
///
///
///
publicstaticListToList(thisDataTabledataTable)whereT:class,new()
{
if(dataTable==null||dataTable.Rows.Count<=0)thrownewArgumentNullException("dataTable","当前对象为null无法生成表达式树");
Funcfunc=dataTable.Rows[0].ToExpression();
Listcollection=newList(dataTable.Rows.Count);
foreach(DataRowdrindataTable.Rows)
{
collection.Add(func(dr));
}
returncollection;
}
///
///生成表达式
///
///
///
///
publicstaticFuncToExpression(thisDataRowdataRow)whereT:class,new()
{
if(dataRow==null)thrownewArgumentNullException("dataRow","当前对象为null无法转换成实体");
ParameterExpressionparamter=Expression.Parameter(typeof(DataRow),"dr");
Listbinds=newList();
for(inti=0;i>(init,paramter).Compile();
}
}
经过测试,用这个方法在同样的条件下转换实体需要0.47秒。除了第一次用反射生成Lambda表达式外,后续的转换直接用的表达式。
以上所述是小编给大家介绍的C#中DataTable转实体实例详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!