ToDataTable<T>思考

在现实的编程当中常会有要把一个IEnumerable接口的数据结构转换成一个DataTable的操作。

 

在C#3.0 Linq以前,我们可能多数会想到自己去写这样一个函数。今天看到国外一牛人些的一篇Blog实现了这样一个功能。

我把其中主要的封装代码复制下来以供学习之用:

代码
 public static class DataSetLinqOperators
    {
        
public static DataTable CopyToDataTable<T>(this IEnumerable<T> source)
        {
            
return new ObjectShredder<T>().Shred(source, nullnull);
        }

        
public static DataTable CopyToDataTable<T>(this IEnumerable<T> source, 
                                                    DataTable table, LoadOption
? options)
        {
            
return new ObjectShredder<T>().Shred(source, table, options);
        }

    }

    
public class ObjectShredder<T>
    {
        
private FieldInfo[] _fi;
        
private PropertyInfo[] _pi;
        
private Dictionary<stringint> _ordinalMap;
        
private Type _type;

        
public ObjectShredder()
        {
            _type 
= typeof(T);
            _fi 
= _type.GetFields();
            _pi 
= _type.GetProperties();
            _ordinalMap 
= new Dictionary<stringint>();
        }

        
public DataTable Shred(IEnumerable<T> source, DataTable table, LoadOption? options)
        {
            
if (typeof(T).IsPrimitive)
            {
                
return ShredPrimitive(source, table, options);   
            }
    

            
if (table == null)
            {
                table 
= new DataTable(typeof(T).Name);
            }

            
// now see if need to extend datatable base on the type T + build ordinal map
            table = ExtendTable(table, typeof(T));

            table.BeginLoadData();
            
using (IEnumerator<T> e = source.GetEnumerator())
            {
                
while (e.MoveNext())
                {
                    
if (options != null)
                    {
                        table.LoadDataRow(ShredObject(table, e.Current), (LoadOption)options);
                    }
                    
else
                    {
                        table.LoadDataRow(ShredObject(table, e.Current), 
true);
                    }
                }
            }
            table.EndLoadData();
            
return table;
        }

        
public DataTable ShredPrimitive(IEnumerable<T> source, DataTable table, LoadOption? options)
        {
            
if (table == null)
            {
                table 
= new DataTable(typeof(T).Name);
            }

            
if (!table.Columns.Contains("Value"))
            {
                table.Columns.Add(
"Value"typeof(T));
            }

            table.BeginLoadData();
            
using (IEnumerator<T> e = source.GetEnumerator())
            {
                Object[] values 
= new object[table.Columns.Count];
                
while (e.MoveNext())
                {
                    values[table.Columns[
"Value"].Ordinal] = e.Current;

                    
if (options != null)
                    {
                        table.LoadDataRow(values, (LoadOption)options);
                    }
                    
else
                    {
                        table.LoadDataRow(values, 
true);
                    }
                }
            }
            table.EndLoadData();  
            
return table; 
        }

        
public DataTable ExtendTable(DataTable table, Type type)
        {
            
// value is type derived from T, may need to extend table.
            foreach (FieldInfo f in type.GetFields())
            {
                
if (!_ordinalMap.ContainsKey(f.Name))
                {
                    DataColumn dc 
= table.Columns.Contains(f.Name) ? table.Columns[f.Name]
                        : table.Columns.Add(f.Name, f.FieldType);
                    _ordinalMap.Add(f.Name, dc.Ordinal);               
                }
            }
            
foreach (PropertyInfo p in type.GetProperties())
            {
                
if (!_ordinalMap.ContainsKey(p.Name))
                {
                    DataColumn dc 
= table.Columns.Contains(p.Name) ? table.Columns[p.Name]
                        : table.Columns.Add(p.Name, p.PropertyType);
                    _ordinalMap.Add(p.Name, dc.Ordinal);
                }
            }
            
return table;
        }

        
public object[] ShredObject(DataTable table, T instance)
        {

            FieldInfo[] fi 
= _fi;
            PropertyInfo[] pi 
= _pi;

            
if (instance.GetType() != typeof(T))
            {
                ExtendTable(table, instance.GetType());
                fi 
= instance.GetType().GetFields();
                pi 
= instance.GetType().GetProperties();
            }

            Object[] values 
= new object[table.Columns.Count];
            
foreach (FieldInfo f in fi)
            {
                values[_ordinalMap[f.Name]] 
= f.GetValue(instance);
            }

            
foreach (PropertyInfo p in pi)
            {
                values[_ordinalMap[p.Name]] 
= p.GetValue(instance, null);
            }
            
return values;
        }
    }


 简单的调用代码如下:

代码
 class Sample
    {
        
static void Main(string[] args)
        {
            
// create sequence 
            Item[] items = new Item[] { new Book{Id = 1, Price = 13.50, Genre = "Comedy", Author = "Jim Bob"}, 
                                        
new Book{Id = 2, Price = 8.50, Genre = "Drama", Author = "John Fox"},  
                                        
new Movie{Id = 1, Price = 22.99, Genre = "Comedy", Director = "Phil Funk"},
                                        
new Movie{Id = 1, Price = 13.40, Genre = "Action", Director = "Eddie Jones"}};

                        
            var query1 
= from i in items
                         
where i.Price > 9.99
                         orderby i.Price
                         select i;

            
// load into new DataTable
            DataTable table1 = query1.CopyToDataTable();

            
// load into existing DataTable - schemas match            
            DataTable table2 = new DataTable();
            table2.Columns.Add(
"Price"typeof(int));
            table2.Columns.Add(
"Genre"typeof(string));

            var query2 
= from i in items
                         
where i.Price > 9.99
                         orderby i.Price
                         select 
new {i.Price, i.Genre};

            query2.CopyToDataTable(table2, LoadOption.PreserveChanges);


            
// load into existing DataTable - expand schema + autogenerate new Id.
            DataTable table3 = new DataTable();
            DataColumn dc 
= table3.Columns.Add("NewId"typeof(int));
            dc.AutoIncrement 
= true;
            table3.Columns.Add(
"ExtraColumn"typeof(string));

            var query3 
= from i in items
                         
where i.Price > 9.99
                         orderby i.Price
                         select 
new { i.Price, i.Genre };

            query3.CopyToDataTable(table3, LoadOption.PreserveChanges);

            
// load sequence of scalars.

            var query4 
= from i in items
                         
where i.Price > 9.99
                         orderby i.Price
                         select i.Price;

            var DataTable4 
= query4.CopyToDataTable();
        }

        
public class Item
        {
            
public int Id { getset; }
            
public double Price { getset; }
            
public string Genre { getset; }   
        }

        
public class Book : Item
        {
            
public string Author { getset; }
        }

        
public class Movie : Item
        {
            
public string Director { getset; }
        }
        
    }


 

 



posted @ 2010-01-21 21:52  卜海清  阅读(548)  评论(0编辑  收藏  举报