MapperExpressionVisitor.cs 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. using Masuit.Tools.Mapping.Helper;
  2. using System.Collections.Generic;
  3. using System.Linq.Expressions;
  4. namespace Masuit.Tools.Mapping.Visitor
  5. {
  6. /// <summary>
  7. /// mapper表达式树访问器
  8. /// </summary>
  9. public class MapperExpressionVisitor : ExpressionVisitor
  10. {
  11. private bool _checkNull;
  12. readonly Stack<MemberExpression> _membersToCheck;
  13. internal Expression Parameter { get; }
  14. /// <summary>
  15. /// 构造函数
  16. /// </summary>
  17. /// <param name="paramClassSource"></param>
  18. public MapperExpressionVisitor(Expression paramClassSource)
  19. {
  20. Parameter = paramClassSource;
  21. _membersToCheck = new Stack<MemberExpression>();
  22. }
  23. /// <summary>
  24. /// 将表达式分布在此类中更专一的访问方法之一。
  25. /// </summary>
  26. /// <param name="node">访问表达式树</param>
  27. /// <param name="checkIfNullity">检查null值</param>
  28. /// <returns>
  29. /// 改变表达式,如果它或它的任何子表达式被修改; 否则,返回原始表达式。
  30. /// </returns>
  31. public Expression Visit(Expression node, bool checkIfNullity = false)
  32. {
  33. _checkNull = checkIfNullity;
  34. if (node == null)
  35. {
  36. return node;
  37. }
  38. if (_checkNull)
  39. {
  40. Expression result;
  41. switch (node.NodeType)
  42. {
  43. case ExpressionType.MemberAccess:
  44. result = VisitMember(node as MemberExpression);
  45. break;
  46. case ExpressionType.Parameter:
  47. result = VisitParameter(node as ParameterExpression);
  48. break;
  49. case ExpressionType.Convert:
  50. result = VisitMember((node as UnaryExpression).Operand as MemberExpression);
  51. break;
  52. case ExpressionType.Lambda:
  53. LambdaExpression lambda = ((LambdaExpression)node);
  54. // 子表达式树
  55. if (lambda.Body.NodeType != ExpressionType.Lambda)
  56. {
  57. result = Visit(lambda.Body);
  58. }
  59. else
  60. {
  61. return lambda;
  62. }
  63. break;
  64. default:
  65. result = base.Visit(node);
  66. break;
  67. }
  68. bool isFirst = true;
  69. Expression previousExpression = null;
  70. if (_membersToCheck.Count > 1)
  71. {
  72. // 在分配值之前测试所有子对象。例如:source.SubClass.SubClass2.MyProperty。
  73. foreach (MemberExpression item in _membersToCheck)
  74. {
  75. if (!isFirst) // 不要测试该属性的值。
  76. {
  77. object defaultValue = MapperHelper.GetDefaultValue(item.Type);
  78. // 创建默认值的验证。
  79. Expression notDefaultValue = Expression.NotEqual(item, Expression.Constant(defaultValue, item.Type));
  80. Expression conditional = null;
  81. // 它创建了包含上述条件的条件。
  82. if (previousExpression != null)
  83. {
  84. object defaultPreviousValue = MapperHelper.GetDefaultValue(previousExpression.Type);
  85. conditional = Expression.Condition(notDefaultValue, previousExpression, Expression.Constant(defaultPreviousValue, previousExpression.Type));
  86. }
  87. // 它会影响新创建的条件,这些条件将成为之前的条件。
  88. previousExpression = conditional;
  89. }
  90. else // 属性
  91. {
  92. previousExpression = item;
  93. isFirst = false;
  94. }
  95. }
  96. return previousExpression;
  97. }
  98. // 不需要递归的元素。
  99. if (_membersToCheck.Count == 1)
  100. {
  101. var item = _membersToCheck.Peek();
  102. object defaultValue = MapperHelper.GetDefaultValue(item.Type);
  103. // 创建默认值的验证。
  104. Expression notDefaultValue = Expression.NotEqual(item, Expression.Constant(defaultValue, item.Type));
  105. Expression conditional = Expression.Condition(notDefaultValue, item, Expression.Constant(defaultValue, item.Type));
  106. return conditional;
  107. }
  108. return result;
  109. }
  110. // 默认返回(更改参数),删除lambda表达式的验证。
  111. if (node.NodeType == ExpressionType.Lambda)
  112. {
  113. LambdaExpression lambda = (LambdaExpression)node;
  114. // 子表达式树
  115. if (lambda.Body.NodeType != ExpressionType.Call)
  116. {
  117. return base.Visit(lambda.Body);
  118. }
  119. return lambda;
  120. }
  121. return base.Visit(node);
  122. }
  123. /// <summary>
  124. /// 访问表达式树
  125. /// </summary>
  126. /// <param name="node">表达式树节点</param>
  127. /// <returns>
  128. /// 改变表达式,如果它或它的任何子表达式被修改; 否则,返回原始表达式。
  129. /// </returns>
  130. protected override Expression VisitParameter(ParameterExpression node)
  131. {
  132. return Parameter;
  133. }
  134. /// <summary>
  135. /// 访问子表达式树
  136. /// </summary>
  137. /// <param name="node">表达式树节点</param>
  138. /// <returns>
  139. /// 改变表达式,如果它或它的任何子表达式被修改; 否则,返回原始表达式。
  140. /// </returns>
  141. protected override Expression VisitMember(MemberExpression node)
  142. {
  143. if (node == null)
  144. {
  145. return node;
  146. }
  147. if (node.Member.ReflectedType != Parameter.Type)
  148. {
  149. var exp = Visit(node.Expression);
  150. return Expression.MakeMemberAccess(exp, node.Member);
  151. }
  152. MemberExpression memberAccessExpression = (MemberExpression)base.VisitMember(node);
  153. // 稍后处理
  154. if (memberAccessExpression != null && _checkNull)
  155. {
  156. // 如果最后一个成员是第一次访问,那么每次都得回去,当前的插入成员位于列表的第一行以更改顺序。
  157. _membersToCheck.Push(memberAccessExpression);
  158. }
  159. return memberAccessExpression;
  160. }
  161. /// <summary>
  162. /// 访问子表达式树
  163. /// </summary>
  164. /// <param name="node">表达式树节点</param>
  165. /// <returns>
  166. /// 改变表达式,如果它或它的任何子表达式被修改; 否则,返回原始表达式。
  167. /// </returns>
  168. protected override Expression VisitUnary(UnaryExpression node)
  169. {
  170. if (node != null)
  171. {
  172. if (node.Operand.NodeType == ExpressionType.MemberAccess)
  173. {
  174. return VisitMember(node.Operand as MemberExpression);
  175. }
  176. if (node.NodeType == ExpressionType.Convert)
  177. {
  178. return Visit(node.Operand);
  179. }
  180. }
  181. return node;
  182. }
  183. }
  184. }