MapperExpressionVisitor.cs 7.7 KB

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