之前都是看别人写博客,自己没有写博客的习惯.在工作的过程中,总是会碰到许多的技术问题.有很多时候想记录下来,后面一直有许多的问题等着解决.总想着等系统完成了,再回头总结下.往往结果就把这事抛到脑后了.
总觉得不能一直这样哈.今天简单记一下吧.有表达不清楚的地方,多多包涵.
最近在研究.net orm框架.想开发一套更好用的Orm框架.碰到一个Expression合并的问题.别嫌轮子多.
一.需求情况描述
需要更新部分数据的时候,可能前端传回的只有部分字段的数据.而更新的时候,需要设置更新人,更新日期等.
举个栗子来说:
现在有一个预约信息表
前端需要修改数据内容如下,我们暂且叫表达A
var exp = ExpressionHelper.CreateExpression<AppointmentDto>(a => new {
a.Id,
a.PatientName,
a.PatientNamePy,
a.IdentityCardNumber,
a.Birthday,
a.PatientAge,
a.PatientSex,
a.PatientPhone,
a.Address
});
而写入数据库的时候需要添加更新人,更新时间.LastUpdateUserId和UpdateTime.
于是我们便又多了一个lambda表达式,我们叫它表达式B
var exp = ExpressionHelper.CreateExpression<AppointmentDto>(a => new {
a.Id,
a.PatientName,
a.PatientNamePy,
a.IdentityCardNumber,
a.Birthday,
a.PatientAge,
a.PatientSex,
a.PatientPhone,
a.Address,
a.LastUpdateUserId,
a.UpdateTime
});
这里说下ExpressionHelper.CreateExpression<T>方法,只是一个为了缩减代码长度而写的方法.输入的lambda表达式原样返回了.
外面不用写好长的类型了.Expression这个类型平时不用.写外面看着眼晕. Expression<Func<AppointmentDto, object>> exp1 = a => new {a.Id,a.PatientName};
/// <summary>
/// 转换Expr
/// 在外面调用时可以使用var以减少代码长度
/// </summary>
/// <param name="expr"></param>
/// <returns></returns>
public static Expression<Func<T, object>> CreateExpression<T>(Expression<Func<T, object>> expr)
{
return expr;
}
所以此处,使用var可以看起来更整洁.但并不推荐在正常情况下使用var.
个人觉得使用var让代码可维护性降低.读起来真的是头疼.之前在维护一个比较大的系统的时候,公司的主要项目,缺少项目文档,代码里面也基本上没啥注释.而且又清一色的var,每个方法返回的是啥类型?你得去方法那边看去.看着真是恼火,又不得不去一点一点的改.都改成相应的类型后,看着就清爽多了.看一眼,流程就基本上能明白大概.所以,var在C#这种强类型语言里,能不用就别用了.
上面就当是发牢骚了.我们回到正题.
我们看到表达式B比表达式A只多了两个字段.大多数代码都是重复的.而且,两个lambda表达式严重的加长了代码行数.几个这样的表达式下来,这个类就到了几百行了.
对于喜欢简洁,简单的我来说,类一大了我就头疼.那咋整?要是有办法将这两个表达式简化处理一下就好了.将表达式A加上一个短的表达式,来实现表达式B呢.
比如实现 var exprB = exprA.Add(a => new { a.PatientPhone });
So,开始捯饬...
二.解决方法
因为这个合并表达式的方法是在个人系统内部使用满足我定制的Orm的类名称需求
所以定义了一个新的Expression表达式类型NewObjectExpression来处理
1 /// <summary> 2 /// New Object Expression 3 /// 合并NewExpression使用. 4 /// </summary> 5 public class NewObjectExpression : Expression, IArgumentProvider 6 { 7 private IList<Expression> arguments; 8 9 /// <summary> 10 /// 构造方法 11 /// </summary> 12 /// <param name="constructor"></param> 13 /// <param name="arguments"></param> 14 /// <param name="members"></param> 15 internal NewObjectExpression(ConstructorInfo constructor, IList<Expression> arguments, List<MemberInfo> members) 16 { 17 this.Constructor = constructor; 18 this.arguments = arguments; 19 this.Members = members; 20 21 if (members != null) 22 { 23 List<string> nameList = members.Select(member => member.Name).ToList(); 24 for (int i = 0; i < nameList.Count; i++) 25 { 26 if (!string.IsNullOrEmpty(ExpressionString)) 27 { 28 ExpressionString += "," + nameList[i]; 29 } 30 else 31 { 32 ExpressionString = nameList[i]; 33 } 34 } 35 } 36 } 37 38 /// <summary> 39 /// Gets the static type of the expression that this <see cref="Expression" /> represents. (Inherited from <see cref="Expression"/>.) 40 /// </summary> 41 /// <returns>The <see cref="Type"/> that represents the static type of the expression.</returns> 42 public override Type Type 43 { 44 get { return Constructor.DeclaringType; } 45 } 46 47 /// <summary> 48 /// Returns the node type of this <see cref="Expression" />. (Inherited from <see cref="Expression" />.) 49 /// </summary> 50 /// <returns>The <see cref="ExpressionType"/> that represents this expression.</returns> 51 public sealed override ExpressionType NodeType 52 { 53 get { return ExpressionType.New; } 54 } 55 56 /// <summary> 57 /// Gets the called constructor. 58 /// </summary> 59 public ConstructorInfo Constructor { get; } 60 61 /// <summary> 62 /// Gets the arguments to the constructor. 63 /// </summary> 64 public ReadOnlyCollection<Expression> Arguments 65 { 66 get { return (ReadOnlyCollection<Expression>)arguments; } 67 } 68 69 Expression IArgumentProvider.GetArgument(int index) 70 { 71 return arguments[index]; 72 } 73 74 int IArgumentProvider.ArgumentCount 75 { 76 get 77 { 78 return arguments.Count; 79 } 80 } 81 82 /// <summary> 83 /// ExpressionString 84 /// </summary> 85 public string ExpressionString { get; private set; } = ""; 86 87 public ConstructorInfo Constructor1 => Constructor; 88 89 public List<MemberInfo> Members { get; set; } 90 91 /// <summary> 92 /// 更新members 93 /// </summary> 94 /// <param name="arguments"></param> 95 /// <param name="members"></param> 96 /// <returns></returns> 97 public NewObjectExpression Update(IList<Expression> arguments, List<MemberInfo> members) 98 { 99 if (arguments != null) 100 { 101 this.arguments = arguments; 102 } 103 if (Members != null) 104 { 105 this.Members = members; 106 ExpressionString = ""; 107 List<string> nameList = members.Select(member => member.Name).ToList(); 108 for (int i = 0; i < nameList.Count; i++) 109 { 110 if (!string.IsNullOrEmpty(ExpressionString)) 111 { 112 ExpressionString += "," + nameList[i]; 113 } 114 else 115 { 116 ExpressionString = nameList[i]; 117 } 118 } 119 } 120 return this; 121 } 122 }