背景:一个实体类字段超级多, 且无法满足使用字典或字符串拼接表示的数据。同时有一定规则,比如 计算某某天占比,总数。例如:
/// <summary> /// 一天不合格 台数 ///</summary> public int Notconforming1Count { get; set; } /// <summary> /// 一天不合格 百分比 ///</summary> public double Notconforming1Percent { get; set; } //等等。。。 如果一直要统计 一周,一月,一季度, 一年。。。
特别说明:命名空间 System.Linq 的 Enumerable类中 有很多为实现了 IEnumerable 接口 的类 提供的 扩展方法 排序 分组 求和 求均值 等等等等 很多很多。 在文章 第二部分 就是因为要使用 Sum扩展方法 第二个参数 为 Func<TSource, double> selector ,目的就是要生成该参数。
大致步骤如下
var propName_Count = $"Notconforming{(7 - i).ToStr()}Count"; //规则根据自己业务来定
var propName_Percent = $"Notconforming{(7 - i).ToStr()}Percent";
var propertyInfo_Count = entityType .GetProperty(propName_Count);
var propertyInfo_Percent = entityType .GetProperty(propName_Percent);
封装方法为:
/// <summary> /// 根据属性名称 生成具体的 实际的 委托 /// </summary> /// <typeparam name="TEntity"></typeparam> /// <param name="propertyName"></param> /// <returns></returns> private static LambdaExpression CreatePropertyGetterExpression<TEntity>(string propertyName) { var entityType = typeof(TEntity); PropertyInfo property = entityType.GetProperty(propertyName); var parameter = Expression.Parameter(entityType, "e"); var body = Expression.MakeMemberAccess(parameter, property); return Expression.Lambda(body, parameter); }
使用方式:
var myDelegate CreatePropertyGetterExpression<你的实体类型>(propName_Percent).Compile(); //propName_Percent 为属性名称 字符串表示
//CreatePropertyGetterExpression根据传入的类型 与 属性名称 生成 LambdaExpression 表达式 对象
//调用Compile 生成表示 lambda 表达式的委托。
var actualDelegate = (Func<你的实体类型, double>)myDelegate; //doublue 为 Tresult中的返回值类型 根据你的属性决定。
此时 actualDelegate 即为期望的参数。
完整示例如下: