我有以下两个表达式:
Expression<Func<string, bool>> expr1 = s => s.Length == 5;
Expression<Func<string, bool>> expr2 = s => s == "someString";
现在我需要将它们与OR结合起来。像这样:
Expression.Or(expr1, expr2)
有没有办法让它类似于上面的代码方式:
expr1 || expr2
我理解在这个例子中我可以把它组合起来:
Expression<Func<string, bool>> expr = s => s.Length == 5 || s == "someString"
但我无法在我的真实代码中执行此操作,因为我将expr1和expr2作为方法的参数。
答案 0 :(得分:52)
使用.NET 4中引入的新ExpressionVisitor
而不是自定义重写器来完成Eric的答案:
internal class ParameterReplacer : ExpressionVisitor {
private readonly ParameterExpression _parameter;
protected override Expression VisitParameter(ParameterExpression node) {
return base.VisitParameter(_parameter);
}
internal ParameterReplacer(ParameterExpression parameter) {
_parameter = parameter;
}
}
class Program {
static void Main(string[] args) {
Expression<Func<string, bool>> expr1 = s => s.Length == 5;
Expression<Func<string, bool>> expr2 = s => s == "someString";
var paramExpr = Expression.Parameter(typeof(string));
var exprBody = Expression.Or(expr1.Body, expr2.Body);
exprBody = (BinaryExpression) new ParameterReplacer(paramExpr).Visit(exprBody);
var finalExpr = Expression.Lambda<Func<string, bool>>(exprBody, paramExpr);
}
}
答案 1 :(得分:23)
问题是每个lambda中的“s”参数具有相同的名称和相同的类型,但它们是不同的参数。参数具有引用标识,而不是值标识。只需将现有表达式树的两个实体组合成第三个表达式树,即可实现:
s => s1.Length == 5 || s2 == "somestring"
没有任何意义。您要做的是编写一个访问者,使用 new 参数搜索和替换参数s,然后将该参数用作新lambda表达式的参数。
有关详细信息,请参阅此相关问题: