使用Expression代替反射读取IDataReader或IDataRecord给实体类赋值
ExpressionMapper代码
using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Data;
using System.Data.Common;
using System.Linq;
using System.Linq.Expressions;
using System.Reflection;
using System.Text;
using System.Threading.Tasks;
namespace LiteSql
{
    /// <summary>
    /// IDataRecord到实体类的映射
    /// </summary>
    internal class ExpressionMapper
    {
        #region 变量
        /// <summary>
        /// 缓存
        /// </summary>
        private static ConcurrentDictionary<string, object> _cacheDict = new ConcurrentDictionary<string, object>();
        #endregion
        #region BindData<T> 数据绑定
        /// <summary>
        /// 数据绑定
        /// </summary>
        public static Func<IDataRecord, T> BindData<T>(PropertyInfoEx[] propertyInfoList, Dictionary<string, int> fields, Dictionary<string, Type> fieldTypes, string strFields)
        {
            Type entityType = typeof(T);
            string key = entityType.FullName + "_" + strFields + "_T";
            if (_cacheDict.TryGetValue(key, out _))
            {
                return _cacheDict[key] as Func<IDataRecord, T>;
            }
            else
            {
                CreateBindings(entityType, propertyInfoList, fields, fieldTypes, out ParameterExpression dataRecordExpr, out Expression initExpr);
                Expression<Func<IDataRecord, T>> lambda = Expression.Lambda<Func<IDataRecord, T>>(initExpr, dataRecordExpr);
                var func = lambda.Compile();
                _cacheDict.TryAdd(key, func);
                return func;
            }
        }
        #endregion
        #region BindData 数据绑定
        /// <summary>
        /// 数据绑定
        /// </summary>
        public static Func<IDataRecord, object> BindData(Type entityType, PropertyInfoEx[] propertyInfoList, Dictionary<string, int> fields, Dictionary<string, Type> fieldTypes, string strFields)
        {
            string key = entityType.FullName + "_" + strFields + "_Type";
            if (_cacheDict.TryGetValue(key, out _))
            {
                return _cacheDict[key] as Func<IDataRecord, object>;
            }
            else
            {
                CreateBindings(entityType, propertyInfoList, fields, fieldTypes, out ParameterExpression dataRecordExpr, out Expression initExpr);
                Expression<Func<IDataRecord, object>> lambda = Expression.Lambda<Func<IDataRecord, object>>(initExpr, dataRecordExpr);
                var func = lambda.Compile();
                _cacheDict.TryAdd(key, func);
                return func;
            }
        }
        #endregion
        #region CreateBindings 属性绑定
        /// <summary>
        /// 属性绑定
        /// </summary>
        private static void CreateBindings(Type entityType, PropertyInfoEx[] propertyInfoList, Dictionary<string, int> fields, Dictionary<string, Type> fieldTypes, out ParameterExpression dataRecordExpr, out Expression initExpr)
        {
            dataRecordExpr = Expression.Parameter(typeof(IDataRecord), "r");
            List<MemberBinding> bindings = new List<MemberBinding>();
            foreach (PropertyInfoEx propertyInfoEx in propertyInfoList)
            {
                if (!fields.ContainsKey(propertyInfoEx.FieldNameUpper)) continue;
                Expression propertyValue = GetMethodCall(dataRecordExpr, propertyInfoEx, fields[propertyInfoEx.FieldNameUpper], fieldTypes[propertyInfoEx.FieldNameUpper]);
                MemberBinding binding = Expression.Bind(propertyInfoEx.PropertyInfo, propertyValue);
                bindings.Add(binding);
            }
            initExpr = Expression.MemberInit(Expression.New(entityType), bindings);
        }
        #endregion
        #region GetMethodCall
        private static Expression GetMethodCall(ParameterExpression dataRecordExpr, PropertyInfoEx propertyInfoEx, int fieldIndex, Type fieldType)
        {
            Type propertyType = propertyInfoEx.PropertyInfo.PropertyType;
            Type typeIDataRecord = typeof(IDataRecord);
            string methodName = GetIDataRecordMethod(ref fieldType);
            MethodCallExpression getValueExpr = Expression.Call(dataRecordExpr, typeIDataRecord.GetMethod(methodName), Expression.Constant(fieldIndex));
            MethodCallExpression isDBNullExpr = Expression.Call(dataRecordExpr, typeIDataRecord.GetMethod("IsDBNull"), Expression.Constant(fieldIndex));
            var convertExpr = GetConvertExpr(propertyType, fieldType, getValueExpr);
            return Expression.Condition(isDBNullExpr, Expression.Default(propertyType), convertExpr);
        }
        #endregion
        #region GetConvertExpr
        private static Expression GetConvertExpr(Type propertyType, Type fieldType, Expression getValueExpr)
        {
            Expression convertExpr = null;
            Type genericType = null;
            if (propertyType.IsGenericType && propertyType.GetGenericTypeDefinition().Equals(typeof(Nullable<>)))
            {
                genericType = propertyType.GenericTypeArguments[0];
            }
            string methodName = GetConvertMethod(propertyType, out Type methodReturnType);
            if (propertyType != fieldType)
            {
                if (genericType != null)
                {
                    if (genericType == typeof(Guid))
                    {
                        Expression newGuidExpr = Expression.New(typeof(Guid).GetConstructor(new Type[] { typeof(string) }), getValueExpr);
                        convertExpr = Expression.Convert(newGuidExpr, propertyType);
                    }
                    else if (genericType != fieldType)
                    {
                        convertExpr = Expression.Call(typeof(Convert).GetMethod(methodName, new Type[] { fieldType }), getValueExpr);
                        convertExpr = Expression.Convert(convertExpr, propertyType);
                    }
                    else
                    {
                        convertExpr = Expression.Convert(getValueExpr, propertyType);
                    }
                }
                else
                {
                    if (propertyType == typeof(Guid))
                    {
                        Expression newGuidExpr = Expression.New(typeof(Guid).GetConstructor(new Type[] { typeof(string) }), getValueExpr);
                        convertExpr = newGuidExpr;
                    }
                    else if (methodName != null && methodReturnType != fieldType)
                    {
                        convertExpr = Expression.Call(typeof(Convert).GetMethod(methodName, new Type[] { fieldType }), getValueExpr);
                        if (propertyType != methodReturnType)
                        {
                            convertExpr = Expression.Convert(convertExpr, propertyType);
                        }
                    }
                    else
                    {
                        if (propertyType != methodReturnType || propertyType == typeof(byte[]))
                        {
                            convertExpr = Expression.Convert(getValueExpr, propertyType);
                        }
                        else if (propertyType != methodReturnType || propertyType.BaseType == typeof(Array))
                        {
                            convertExpr = Expression.Convert(getValueExpr, propertyType);
                        }
                        else
                        {
                            convertExpr = getValueExpr;
                        }
                    }
                }
            }
            else
            {
                convertExpr = getValueExpr;
            }
            return convertExpr;
        }
        #endregion
        #region GetIDataRecordMethod 获取IDataRecord方法名称和返回值类型
        /// <summary>
        /// 获取IDataRecord方法名称和返回值类型
        /// </summary>
        private static string GetIDataRecordMethod(ref Type fieldType)
        {
            string methodName = "GetValue";
            if (fieldType == typeof(string))
            {
                methodName = "GetString";
            }
            else if (fieldType == typeof(object))
            {
                methodName = "GetValue";
            }
            else if (fieldType == typeof(byte[]))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(Guid))
            {
                methodName = "GetString";
                fieldType = typeof(string);
            }
            else if (fieldType == typeof(char))
            {
                methodName = "GetChar";
            }
            else if (fieldType == typeof(byte))
            {
                methodName = "GetByte";
            }
            else if (fieldType == typeof(sbyte))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(short))
            {
                methodName = "GetInt16";
            }
            else if (fieldType == typeof(ushort))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(int))
            {
                methodName = "GetInt32";
            }
            else if (fieldType == typeof(uint))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(long))
            {
                methodName = "GetInt64";
            }
            else if (fieldType == typeof(ulong))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(float))
            {
                methodName = "GetFloat";
            }
            else if (fieldType == typeof(double))
            {
                methodName = "GetDouble";
            }
            else if (fieldType == typeof(decimal))
            {
                methodName = "GetDecimal";
            }
            else if (fieldType == typeof(bool))
            {
                methodName = "GetBoolean";
            }
            else if (fieldType == typeof(DateTime))
            {
                methodName = "GetDateTime";
            }
            // ======== 以下是可空类型 ================================
            else if (fieldType == typeof(Guid?))
            {
                methodName = "GetString";
                fieldType = typeof(string);
            }
            else if (fieldType == typeof(char?))
            {
                methodName = "GetChar";
                fieldType = typeof(char);
            }
            else if (fieldType == typeof(byte?))
            {
                methodName = "GetByte";
                fieldType = typeof(byte);
            }
            else if (fieldType == typeof(sbyte?))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(short?))
            {
                methodName = "GetInt16";
                fieldType = typeof(short);
            }
            else if (fieldType == typeof(ushort?))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(int?))
            {
                methodName = "GetInt32";
                fieldType = typeof(int);
            }
            else if (fieldType == typeof(uint?))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(long?))
            {
                methodName = "GetInt64";
                fieldType = typeof(long);
            }
            else if (fieldType == typeof(ulong?))
            {
                methodName = "GetValue";
                fieldType = typeof(object);
            }
            else if (fieldType == typeof(float?))
            {
                methodName = "GetFloat";
                fieldType = typeof(float);
            }
            else if (fieldType == typeof(double?))
            {
                methodName = "GetDouble";
                fieldType = typeof(double);
            }
            else if (fieldType == typeof(decimal?))
            {
                methodName = "GetDecimal";
                fieldType = typeof(decimal);
            }
            else if (fieldType == typeof(bool?))
            {
                methodName = "GetBoolean";
                fieldType = typeof(bool);
            }
            else if (fieldType == typeof(DateTime?))
            {
                methodName = "GetDateTime";
                fieldType = typeof(DateTime);
            }
            return methodName;
        }
        #endregion
        #region GetConvertMethod 获取Convert类的方法名称和返回值类型
        /// <summary>
        /// 获取Convert类的方法名称和返回值类型
        /// </summary>
        private static string GetConvertMethod(Type propertyType, out Type methodReturnType)
        {
            string methodName = null;
            methodReturnType = propertyType;
            if (propertyType == typeof(string))
            {
                methodName = "ToString";
            }
            else if (propertyType == typeof(char))
            {
                methodName = "ToChar";
            }
            else if (propertyType == typeof(byte))
            {
                methodName = "ToByte";
            }
            else if (propertyType == typeof(sbyte))
            {
                methodName = "ToByte";
                methodReturnType = typeof(byte);
            }
            else if (propertyType == typeof(short))
            {
                methodName = "ToInt16";
            }
            else if (propertyType == typeof(ushort))
            {
                methodName = "ToInt16";
                methodReturnType = typeof(short);
            }
            else if (propertyType == typeof(int))
            {
                methodName = "ToInt32";
            }
            else if (propertyType == typeof(uint))
            {
                methodName = "ToInt32";
                methodReturnType = typeof(int);
            }
            else if (propertyType == typeof(long))
            {
                methodName = "ToInt64";
            }
            else if (propertyType == typeof(ulong))
            {
                methodName = "ToInt64";
                methodReturnType = typeof(long);
            }
            else if (propertyType == typeof(float))
            {
                methodName = "ToSingle";
            }
            else if (propertyType == typeof(double))
            {
                methodName = "ToDouble";
            }
            else if (propertyType == typeof(decimal))
            {
                methodName = "ToDecimal";
            }
            else if (propertyType == typeof(bool))
            {
                methodName = "ToBoolean";
            }
            else if (propertyType == typeof(DateTime))
            {
                methodName = "ToDateTime";
            }
            // ======== 以下是可空类型 ================================
            else if (propertyType == typeof(char?))
            {
                methodName = "ToChar";
            }
            else if (propertyType == typeof(byte?))
            {
                methodName = "ToByte";
            }
            else if (propertyType == typeof(sbyte?))
            {
                methodName = "ToByte";
                methodReturnType = typeof(byte);
            }
            else if (propertyType == typeof(short?))
            {
                methodName = "ToInt16";
            }
            else if (propertyType == typeof(ushort?))
            {
                methodName = "ToInt16";
                methodReturnType = typeof(short);
            }
            else if (propertyType == typeof(int?))
            {
                methodName = "ToInt32";
            }
            else if (propertyType == typeof(uint?))
            {
                methodName = "ToInt32";
                methodReturnType = typeof(int);
            }
            else if (propertyType == typeof(long?))
            {
                methodName = "ToInt64";
            }
            else if (propertyType == typeof(ulong?))
            {
                methodName = "ToInt64";
                methodReturnType = typeof(long);
            }
            else if (propertyType == typeof(float?))
            {
                methodName = "ToSingle";
            }
            else if (propertyType == typeof(double?))
            {
                methodName = "ToDouble";
            }
            else if (propertyType == typeof(decimal?))
            {
                methodName = "ToDecimal";
            }
            else if (propertyType == typeof(bool?))
            {
                methodName = "ToBoolean";
            }
            else if (propertyType == typeof(DateTime?))
            {
                methodName = "ToDateTime";
            }
            return methodName;
        }
        #endregion
    }
}
如何使用
实体类列表赋值
private void DataReaderToList<T>(IDataReader rd, ref List<T> list) where T : new()
{
    if (typeof(T) == typeof(int))
    {
        while (rd.Read())
        {
            list.Add((T)rd[0]);
        }
    }
    else if (typeof(T) == typeof(string))
    {
        while (rd.Read())
        {
            list.Add((T)rd[0]);
        }
    }
    else
    {
        PropertyInfoEx[] propertyInfoList = GetEntityProperties(typeof(T));
        int fcnt = rd.FieldCount;
        StringBuilder strFields = new StringBuilder();
        Dictionary<string, int> fields = new Dictionary<string, int>();
        Dictionary<string, Type> fieldTypes = new Dictionary<string, Type>();
        for (int i = 0; i < fcnt; i++)
        {
            string field = rd.GetName(i).ToUpper();
            if (!fields.ContainsKey(field))
            {
                fields.Add(field, i);
                fieldTypes.Add(field, rd.GetFieldType(i));
                strFields.Append("_" + field);
            }
        }
        var func = ExpressionMapper.BindData<T>(propertyInfoList, fields, fieldTypes, strFields.ToString());
        while (rd.Read())
        {
            T obj = func(rd);
            list.Add(obj);
        }
    }
}
实体类赋值
private void DataReaderToEntity(Type type, IDataReader rd, ref object result, ref bool hasValue)
{
    PropertyInfoEx[] propertyInfoList = GetEntityProperties(type);
    int fieldCount = rd.FieldCount;
    StringBuilder strFields = new StringBuilder();
    Dictionary<string, int> fields = new Dictionary<string, int>();
    Dictionary<string, Type> fieldTypes = new Dictionary<string, Type>();
    for (int i = 0; i < fieldCount; i++)
    {
        string field = rd.GetName(i).ToUpper();
        if (!fields.ContainsKey(field))
        {
            fields.Add(field, i);
            fieldTypes.Add(field, rd.GetFieldType(i));
            strFields.Append("_" + field);
        }
    }
    var func = ExpressionMapper.BindData(type, propertyInfoList, fields, fieldTypes, strFields.ToString());
    while (rd.Read())
    {
        hasValue = true;
        result = func(rd);
    }
}
使用Expression代替反射读取IDataReader或IDataRecord给实体类赋值的更多相关文章
- 读取Excel文件存储在实体类中
		1.Maven文件 <!--读取Excel的架包--> <dependency> <groupId>org.apache.poi</groupId> & ... 
- [转]C#反射,根据反射将数据库查询数据和实体类绑定,并未实体类赋值
		本文来自:http://www.cnblogs.com/mrchenzh/archive/2010/05/31/1747937.html /****************************** ... 
- .net 通过反射实现两个相同结构实体类的转换
		public static T2 CopyToModel<T1, T2>(T1 source) { T2 model = default(T2); PropertyInfo[] pi = ... 
- .Net Core NPOI读取Excel 并转为数据实体类
		创建应用程序 这里直接创建Console程序 引用NPOI的NuGet包 PM> Install-Package NPOI -Version 2.5.1 直接Nuget包管理器添加 导入Exce ... 
- 转 利用java反射实现两个具有相同属性bean赋值
		package com.dobn.bdgcgl.utils; import java.lang.reflect.Field; import java.lang.reflect.Method; publ ... 
- Android利用反射机制为实体类属性赋值
		在做android项目时,有时会遇到从网络上获取json类型数据,赋值给实体类,实体类属性少可以一个一个的赋值,如果实体类有很多属性,赋值可能就要耗很长的功夫了,幸好Java给我们提供了反射机制.下面 ... 
- 运用Mono.Cecil 反射读取.NET程序集元数据
		CLR自带的反射机智和API可以很轻松的读取.NET程序集信息,但是不能对程序集进行修改.CLR提供的是只读的API,但是开源项目Mono.Cecil不仅仅可以读取.NET程序集的元数据,还可以进行修 ... 
- SqlHelper简单实现(通过Expression和反射)1.引言
		之前老大说要改变代码中充斥着各种Select的Sql语句字符串的情况,让我尝试着做一个简单的SqlHelper,要具有以下功能: 1.不要在业务代码中暴露DataTable或者DataSet类型: 2 ... 
- Java通过反射读取泛型
		1.在这里有必要先提一句关于javabean,建议有一个构造器,而不是必须要写的,这是一个良好的习惯. 这样做肯定是有好处的,如果你的一个类写了带参的构造方法,而没有写空的构造方法,那么,如有有一个类 ... 
- 利用c#反射实现实体类生成以及数据获取与赋值
		转:http://hi.baidu.com/xyd21c/item/391da2fc8fb351c10dd1c8b8 原有的实体类成员逐个赋值与获取的方法弊端: 1.每次对实体类属性进行赋值时,都要检 ... 
随机推荐
- rust程序设计(5)结构体相关练习题| 附带解答
			题目 基础结构体练习: 创建一个名为Person的结构体,包含name(字符串类型)和age(整数类型)两个字段. 写一个函数,接收一个Person实例作为参数,并打印出这个人的名字和年龄. 结构体方 ... 
- jdk所有版本-自留收藏
			链接:https://pan.baidu.com/s/1NDbEAEbKeh8xFzjEwB8aLg 提取码:0000 
- python循环语句(二)
			Python for循环可以遍历任何序列的项目,如一个列表或者一个字符串. 知识点:for循环的语法格式如下: for iterating_var in sequence: statements(s) ... 
- SpringBoot实战项目:蚂蚁爱购(从零开发)
			 简介 这是从零开发的SpringBoot实战项目,名字叫蚂蚁爱购. 从零开发项目,视频加文档,十天彻底掌握开发SpringBoot项目. 教程路线是:搭建环境=> 安装软件=> 创建项 ... 
- [清华集训2017] Hello World!
			Hello world! 题目背景 不远的一年前,小 V 还是一名清华集训的选手,坐在机房里为他已如风中残烛的OI 生涯做最后的挣扎.而如今,他已成为了一名光荣的出题人.他感到非常激动,不禁感叹道: ... 
- Javascript实现Canvas绘图 —— 2D绘图之填充、描边及绘制矩形
			Canvas绘图的实现: <canvas>元素负责在页面中设定一个区域,通过JS动态地在这个区域中绘制图形. IE9+.Firefox1.5+.Safari2+.Opera9+.Chrom ... 
- Tomcat自动化脚本
			/bin/bash war包名称 war_name="tchg.war" 要上传war包指定目录 war_dir="/usr/local/src/tchg" 工 ... 
- Python——第二章:文件操作
			文件操作 1. 找到这个文件. 双击打开它 open(文件路径, mode="", encoding="") 文件路径: 1. 绝对路径 d:/test/xxx ... 
- 安装Zabbix客户端zabbix-agent2
			Ubuntu系统:下载链接:http://mirrors.aliyun.com/zabbix/zabbix/6.0/ubuntu/pool/main/z/zabbix/?spm=a2c6h.25603 ... 
- 19、Flutter StatelessWidget 、 StatefulWidget
			在Flutter中自定义组件其实就是一个类,这个类需要继承StatelessWidget / StatefulWidget. StatelessWidget是无状态组件,状态不可变的widget ... 
