MapperExpressionVisitor.cs 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  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. // source.SubClass!= null? source.SubClass.SubClass2!= null? source.SubClass.SubClass2.MyProperty:DefaultValueOfProperty:DefaultValueOfProperty
  74. foreach (MemberExpression item in _membersToCheck)
  75. {
  76. if (!isFirst) // 不要测试该属性的值。
  77. {
  78. object defaultValue = MapperHelper.GetDefaultValue(item.Type);
  79. // 创建默认值的验证。
  80. Expression notDefaultValue = Expression.NotEqual(item, Expression.Constant(defaultValue, item.Type));
  81. Expression conditional = null;
  82. // 它创建了包含上述条件的条件。
  83. if (previousExpression != null)
  84. {
  85. object defaultPreviousValue = MapperHelper.GetDefaultValue(previousExpression.Type);
  86. conditional = Expression.Condition(notDefaultValue, previousExpression, Expression.Constant(defaultPreviousValue, previousExpression.Type));
  87. }
  88. // 它会影响新创建的条件,这些条件将成为之前的条件。
  89. previousExpression = conditional;
  90. }
  91. else // 属性
  92. {
  93. previousExpression = item;
  94. isFirst = false;
  95. }
  96. }
  97. return previousExpression;
  98. }
  99. // 不需要递归的元素。
  100. if (_membersToCheck.Count == 1)
  101. {
  102. var item = _membersToCheck.Peek();
  103. object defaultValue = MapperHelper.GetDefaultValue(item.Type);
  104. // 创建默认值的验证。
  105. Expression notDefaultValue = Expression.NotEqual(item, Expression.Constant(defaultValue, item.Type));
  106. Expression conditional = Expression.Condition(notDefaultValue, item, Expression.Constant(defaultValue, item.Type));
  107. return conditional;
  108. }
  109. return result;
  110. }
  111. // 默认返回(更改参数),删除lambda表达式的验证。
  112. if ((node.NodeType == ExpressionType.Lambda))
  113. {
  114. LambdaExpression lambda = ((LambdaExpression)node);
  115. // 子表达式树
  116. if (lambda.Body.NodeType != ExpressionType.Call)
  117. {
  118. return base.Visit(lambda.Body);
  119. }
  120. return lambda;
  121. }
  122. return base.Visit(node);
  123. }
  124. /// <summary>
  125. /// 访问表达式树
  126. /// </summary>
  127. /// <param name="node">表达式树节点</param>
  128. /// <returns>
  129. /// 改变表达式,如果它或它的任何子表达式被修改; 否则,返回原始表达式。
  130. /// </returns>
  131. protected override Expression VisitParameter(ParameterExpression node)
  132. {
  133. return Parameter;
  134. }
  135. /// <summary>
  136. /// 访问子表达式树
  137. /// </summary>
  138. /// <param name="node">表达式树节点</param>
  139. /// <returns>
  140. /// 改变表达式,如果它或它的任何子表达式被修改; 否则,返回原始表达式。
  141. /// </returns>
  142. protected override Expression VisitMember(MemberExpression node)
  143. {
  144. if (node == null)
  145. {
  146. return node;
  147. }
  148. if (node.Member.ReflectedType != Parameter.Type)
  149. {
  150. var exp = Visit(node.Expression);
  151. return Expression.MakeMemberAccess(exp, node.Member);
  152. }
  153. MemberExpression memberAccessExpression = (MemberExpression)base.VisitMember(node);
  154. // 稍后处理
  155. if (memberAccessExpression != null && _checkNull)
  156. {
  157. // 如果最后一个成员是第一次访问,那么每次都得回去,当前的插入成员位于列表的第一行以更改顺序。
  158. _membersToCheck.Push(memberAccessExpression);
  159. }
  160. return memberAccessExpression;
  161. }
  162. /// <summary>
  163. /// 访问子表达式树
  164. /// </summary>
  165. /// <param name="node">表达式树节点</param>
  166. /// <returns>
  167. /// 改变表达式,如果它或它的任何子表达式被修改; 否则,返回原始表达式。
  168. /// </returns>
  169. protected override Expression VisitUnary(UnaryExpression node)
  170. {
  171. if (node != null)
  172. {
  173. if (node.Operand.NodeType == ExpressionType.MemberAccess)
  174. {
  175. return VisitMember(node.Operand as MemberExpression);
  176. }
  177. if (node.NodeType == ExpressionType.Convert)
  178. {
  179. return Visit(node.Operand);
  180. }
  181. }
  182. return node;
  183. }
  184. }
  185. }