我们在学习Linq的时候经常会想怎么样用Linq查询DataTable, 这个很简单,下面的代码就演示了怎么去做。查询的结果可以是Datarow集合也可以是任意自定义的对象集合。
//Linq 查询 Datatable,返回DataRow的集合
var q = from r in DataTable.AsEnumerable()
where r.Field<int>("ID") > 100 && r.Field<bool>("Male") == false
//可以添加更多的筛选条件
select r;
//Linq查询DataTable,返回People的自定义对象结合
var q = from r in DataTable.AsEnumerable()
where r.Field<int>("ID") > 100 && r.Field<bool>("Male") == false
select new People
{
ID = r.Field<int>("ID"),
Name = r.Field<string>("Name"),
Male= r.Field<bool>("Male")
};
然后我们有时候会反过来用,就是对Object集合用Linq查询,想把查询结果生成DataTable。这种用法不常见,所以文章也很少。这里我们就简单介绍一下如何实现。
1. 首先我们对Object集合用Linq查询之后肯定还是生成Object集合
2. 然后在对筛选出来的Object集合用foreach循环时,把数据装入DataTable
3. 这里有个问题,我们如何自适应定义Datatable的结构 (根据Object的结构创建对应的Datatable结构)。这里就需要用到反射,用反射获取对象的所有属性和对应的数据类型,然后动态生成Datatable结构
//用反射遍历对象的属性和类型,并创建对应的Datatable
DataTable Obj_Table = new DataTable("Object Table");
//ppl是People对象的实例
foreach (PropertyInfo pi in ppl.GetType().GetProperties())
{
Type t = pi.PropertyType;
//这个是用来对于可空类型的属性,获取它的基础类型。比如 "int?" 的基础类型是"Int32"
if (t.IsGenericType && t.GetGenericTypeDefinition() == typeof(Nullable<>))
{
t = t.GetGenericArguments()[0];
}
//如果没有可空类型的属性,直接调用下面的一句即可
Obj_Table.Columns.Add(pi.Name, t);
}
//通过这个foreach循环之后,Datatable就已经建好了,对象的每个属性都有一个与之对应的Column,且ColumnName就是属性名,数据类型也一致
DataTable的结构建好之后,我们就要来Linq查询对象集合,然后把数据存入datatable
//筛选对象集合
var qp = from p in Peoples
where p.ID>10 && p.Male==true
select p;
//把符合条件的对象数据存入Datatable
foreach(var p in qp)
{
DataRow dr = Obj_Table.NewRow();
//对当前对象用反射遍历所有属性,并将属性值存入Datatable
foreach (PropertyInfo pi in p.GetType().GetProperties())
{
object v = pi.GetValue(p, null);
if (v != null)
dr[pi.Name] = v;//属性值存入对应的Column
else
dr[pi.Name] = DBNull.Value;//对象属性值为null时用DBNull代替
}
//当前行加入DataTable
Obj_Table.Rows.Add(dr);
}
总结:这是非常规用法,特俗指出就在于不用考虑对象的结构,通过反射自动获取属性创建Datatable的结构,然后依然是用反射自动给遍历获取对象的值存入data table.