zoukankan      html  css  js  c++  java
  • 使用Emit把Datatable转换为对象集合(List<T>)

      Emit生成动态方法部分摘自网上,但是经过修改,加入了对委托的缓存以及类结构的调整,使之调用更简洁方便。大致的思路是:要实现转换datatable到某个指定对象的集合,本质是实现转换一个datarow到一个指定的对象。利用Emit动态构造该方法并缓存起来,调用时从缓存查找出来并调用,就这么简单。上代码:

     

     /// <summary>
        /// 把datatable转换为对象集合列表List<T>
        /// </summary>
        public class DataTableConvert
        {
            //把DataRow转换为对象的委托声明
            private delegate T Load<T>(DataRow dataRecord);
    
            //用于构造Emit的DataRow中获取字段的方法信息
            private static readonly MethodInfo getValueMethod = typeof(DataRow).GetMethod("get_Item", new Type[] { typeof(int) });
    
            //用于构造Emit的DataRow中判断是否为空行的方法信息
            private static readonly MethodInfo isDBNullMethod = typeof(DataRow).GetMethod("IsNull", new Type[] { typeof(int) });
            
            //使用字典存储实体的类型以及与之对应的Emit生成的转换方法
            private static Dictionary<Type, Delegate> rowMapMethods = new Dictionary<Type, Delegate>();
          
            public static List<T> ToList<T>(DataTable dt)
            {
                List<T> list = new List<T>();
                if (dt == null)
                    return list;
    
                //声明 委托Load<T>的一个实例rowMap
                Load<T> rowMap = null;
    
    
                //从rowMapMethods查找当前T类对应的转换方法,没有则使用Emit构造一个。
                if (!rowMapMethods.ContainsKey(typeof(T)))
                {
                    DynamicMethod method = new DynamicMethod("DynamicCreateEntity_" + typeof(T).Name, typeof(T), new Type[] { typeof(DataRow) }, typeof(T), true);
                    ILGenerator generator = method.GetILGenerator();
                    LocalBuilder result = generator.DeclareLocal(typeof(T));
                    generator.Emit(OpCodes.Newobj, typeof(T).GetConstructor(Type.EmptyTypes));
                    generator.Emit(OpCodes.Stloc, result);
    
                    for (int index = 0; index < dt.Columns.Count; index++)
                    {
                        PropertyInfo propertyInfo = typeof(T).GetProperty(dt.Columns[index].ColumnName,StringComparison.CurrentCultureIgnoreCase);
                        Label endIfLabel = generator.DefineLabel();
                        if (propertyInfo != null && propertyInfo.GetSetMethod() != null)
                        {
                            generator.Emit(OpCodes.Ldarg_0);
                            generator.Emit(OpCodes.Ldc_I4, index);
                            generator.Emit(OpCodes.Callvirt, isDBNullMethod);
                            generator.Emit(OpCodes.Brtrue, endIfLabel);
                            generator.Emit(OpCodes.Ldloc, result);
                            generator.Emit(OpCodes.Ldarg_0);
                            generator.Emit(OpCodes.Ldc_I4, index);
                            generator.Emit(OpCodes.Callvirt, getValueMethod);
                            generator.Emit(OpCodes.Unbox_Any, propertyInfo.PropertyType);
                            generator.Emit(OpCodes.Callvirt, propertyInfo.GetSetMethod());
                            generator.MarkLabel(endIfLabel);
                        }
                    }
                    generator.Emit(OpCodes.Ldloc, result);
                    generator.Emit(OpCodes.Ret);
    
                    //构造完成以后传给rowMap
                    rowMap = (Load<T>)method.CreateDelegate(typeof(Load<T>));
                }
                else
                {
                    rowMap = (Load<T>)rowMapMethods[typeof(T)];
                }
    
                //遍历Datatable的rows集合,调用rowMap把DataRow转换为对象(T)
                foreach (DataRow info in dt.Rows)
                    list.Add(rowMap(info));
                return list;
    
            }
        }

     

  • 相关阅读:
    android基础开发之scrollview
    java网络---再论URL & URI
    Android Studio 有用的插件
    java网络---查找Internet
    java网络---流
    Qt学习之路(1)------Qt常用类用法说明
    将批量下载的博客导入到手机后,通过豆约翰博客阅读器APP(Android手机)进行浏览,白字黑底,保护眼睛,图文并茂。
    如何收藏互联网上的任意网页到系统某个分类下,之后进行批量导出发布等---博客备份专家的博文收藏功能您不可不知
    很喜欢看某方面的文章,如何将不同站点,不同博主同一类别的文章归类整合到一起,再批量导出成各种格式---豆约翰博客备份专家新增按分类收藏博文功能
    豆约翰博客备份专家博客导出示例(PDF,CHM)
  • 原文地址:https://www.cnblogs.com/lindping/p/DatatableToList.html
Copyright © 2011-2022 走看看