36

C# in a Nutshell 有一个名为 PredicateBuilder 的免费类,它可以在此处逐段构造 LINQ 谓词。这是向谓词添加新表达式的方法的摘录。有人可以解释一下吗?(我已经看到了这个问题,我不想要像那里那样的一般性答案。我正在寻找关于 Expression.Invoke 和 Expression.Lambda 如何构建新表达式的具体解释)。

public static Expression<Func<T, bool>> And<T> (this Expression<Func<T, bool>> expr1,
                                                     Expression<Func<T, bool>> expr2)
{
    var invokedExpr = Expression.Invoke (expr2, expr1.Parameters.Cast<Expression> ());
    return Expression.Lambda<Func<T, bool>>
        (Expression.AndAlso (expr1.Body, invokedExpr), expr1.Parameters);
}
4

1 回答 1

54

假设您有:

Expression<Func<Person, bool>> isAdult = p1 => p1.Age >= 18;

// I've given the parameter a different name to allow you to differentiate.
Expression<Func<Person, bool>> isMale = p2 => p2.Gender == "Male";

然后将它们与PredicateBuilder

var isAdultMale = isAdult.And(isMale);

产生的是PredicateBuilder一个看起来像这样的表达式:

// Invoke has no direct equivalent in C# lambda expressions.
p1 => p1.Age >= 18 && Invoke(p2 => p2.Gender == "Male", p1)

如你看到的:

  1. 生成的 lambda重用第一个表达式的参数。
  2. 具有通过传递第一个表达式的参数作为第二个表达式参数的替换来调用第二个表达式的主体。结果InvocationExpression 有点像方法调用的表达式等效项(通过传入参数的参数来调用例程)。
  3. Ands 第一个表达式的主体和 thisInvocationExpression一起生成结果 lambda 的主体。

这个想法是 LINQ 提供者应该能够理解这个操作的语义并采取合理的行动(例如生成类似 SQL WHERE age >= 18 AND gender = 'Male')。

但是,提供者通常会遇到InvocationExpressions 的问题,因为处理“表达式内的嵌套表达式调用”显然很复杂。

为了解决这个问题,LINQKit 还提供了Expand帮助程序。这实质上通过用嵌套表达式的主体替换调用,巧妙地“内联”调用调用,适当地替换嵌套表达式参数的使用(在这种情况下,替换p2p1)。这应该产生类似的东西:

p1 => p1.Age >= 18 && p1.Gender == "Male"

请注意,如果您自己在 lambda 中完成,那么您将如何手动组合这些谓词。但是使用 LINQKit,您可以从独立的来源获取这些谓词并轻松组合它们:

  1. 无需编写“手动”表达式代码。
  2. 可选地,以对结果 lambda 的消费者透明的方式。
于 2012-07-15T12:24:10.373 回答