LinqExtension.cs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq.Expressions;
  4. namespace Masuit.Tools.Linq;
  5. /// <summary>
  6. /// LINQ扩展方法
  7. /// </summary>
  8. public static class LinqExtension
  9. {
  10. /// <summary>
  11. /// 与连接
  12. /// </summary>
  13. /// <typeparam name="T">类型</typeparam>
  14. /// <param name="left">左条件</param>
  15. /// <param name="right">右条件</param>
  16. /// <returns>新表达式</returns>
  17. public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> left, Expression<Func<T, bool>> right)
  18. {
  19. return CombineLambdas(left, right, ExpressionType.AndAlso);
  20. }
  21. /// <summary>
  22. /// 与连接
  23. /// </summary>
  24. /// <typeparam name="T">类型</typeparam>
  25. /// <param name="left">左条件</param>
  26. /// <param name="condition">符合什么条件</param>
  27. /// <param name="right">右条件</param>
  28. /// <returns>新表达式</returns>
  29. public static Expression<Func<T, bool>> AndIf<T>(this Expression<Func<T, bool>> left, bool condition, Expression<Func<T, bool>> right)
  30. {
  31. return condition ? CombineLambdas(left, right, ExpressionType.AndAlso) : left;
  32. }
  33. /// <summary>
  34. /// 与连接
  35. /// </summary>
  36. /// <typeparam name="T">类型</typeparam>
  37. /// <param name="left">左条件</param>
  38. /// <param name="condition">符合什么条件</param>
  39. /// <param name="right">右条件</param>
  40. /// <returns>新表达式</returns>
  41. public static Expression<Func<T, bool>> AndIf<T>(this Expression<Func<T, bool>> left, Func<bool> condition, Expression<Func<T, bool>> right)
  42. {
  43. return condition() ? CombineLambdas(left, right, ExpressionType.AndAlso) : left;
  44. }
  45. /// <summary>
  46. /// 或连接
  47. /// </summary>
  48. /// <typeparam name="T">类型</typeparam>
  49. /// <param name="left">左条件</param>
  50. /// <param name="right">右条件</param>
  51. /// <returns>新表达式</returns>
  52. public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> left, Expression<Func<T, bool>> right)
  53. {
  54. return CombineLambdas(left, right, ExpressionType.OrElse);
  55. }
  56. /// <summary>
  57. /// 或连接
  58. /// </summary>
  59. /// <typeparam name="T">类型</typeparam>
  60. /// <param name="left">左条件</param>
  61. /// <param name="condition">符合什么条件</param>
  62. /// <param name="right">右条件</param>
  63. /// <returns>新表达式</returns>
  64. public static Expression<Func<T, bool>> OrIf<T>(this Expression<Func<T, bool>> left, bool condition, Expression<Func<T, bool>> right)
  65. {
  66. return condition ? CombineLambdas(left, right, ExpressionType.OrElse) : left;
  67. }
  68. /// <summary>
  69. /// 或连接
  70. /// </summary>
  71. /// <typeparam name="T">类型</typeparam>
  72. /// <param name="left">左条件</param>
  73. /// <param name="condition">符合什么条件</param>
  74. /// <param name="right">右条件</param>
  75. /// <returns>新表达式</returns>
  76. public static Expression<Func<T, bool>> OrIf<T>(this Expression<Func<T, bool>> left, Func<bool> condition, Expression<Func<T, bool>> right)
  77. {
  78. return condition() ? CombineLambdas(left, right, ExpressionType.OrElse) : left;
  79. }
  80. private static Expression<Func<T, bool>> CombineLambdas<T>(this Expression<Func<T, bool>> left, Expression<Func<T, bool>> right, ExpressionType expressionType)
  81. {
  82. var visitor = new SubstituteParameterVisitor
  83. {
  84. Sub =
  85. {
  86. [right.Parameters[0]] = left.Parameters[0]
  87. }
  88. };
  89. Expression body = Expression.MakeBinary(expressionType, left.Body, visitor.Visit(right.Body));
  90. return Expression.Lambda<Func<T, bool>>(body, left.Parameters[0]);
  91. }
  92. }
  93. internal class SubstituteParameterVisitor : ExpressionVisitor
  94. {
  95. public Dictionary<Expression, Expression> Sub = new();
  96. protected override Expression VisitParameter(ParameterExpression node)
  97. {
  98. return Sub.TryGetValue(node, out var newValue) ? newValue : node;
  99. }
  100. }