如何在不使用Invoke的情况下合并两个C# Lambda表达式?

14

我想将以下表达式合并:

// example class
class Order
{
    List<OrderLine> Lines       
}
class OrderLine { }

Expression<Func<Order, List<OrderLine>>> selectOrderLines = o => o.Lines;
Expression<Func<List<OrderLine>, Boolean>> validateOrderLines = lines => lines.Count > 0;

// now combine those to
Expression<Func<Order, Boolean>> validateOrder;

我使用在selectOrderLines上调用并将结果提供给validateOrderLines的方法使其工作,但由于我在Entity Framework中使用这些表达式,我必须实际创建一个干净的表达式来代表:

Expression<Func<Order, Boolean>> validateOrder = o => o.Lines.Count > 0;

我该怎么做?

2个回答

25
最优雅的方法是使用表达式访问器。特别地,这个MSDN博客文章描述了如何使用它来组合谓词(使用布尔And或Or)而不需要Invoke。 编辑:意识到布尔组合不是您想要的,我编写了一个使用ExpressionVisitor解决您特定问题的示例用法:
public class ParameterToMemberExpressionRebinder : ExpressionVisitor
{
    ParameterExpression _paramExpr;
    MemberExpression _memberExpr;

    ParameterToMemberExpressionRebinder(ParameterExpression paramExpr, MemberExpression memberExpr) 
    {
        _paramExpr = paramExpr;
        _memberExpr = memberExpr;
    }

    protected override Expression Visit(Expression p)
    {
        return base.Visit(p == _paramExpr ? _memberExpr : p);
    }

    public static Expression<Func<T, bool>> CombinePropertySelectorWithPredicate<T, T2>(
        Expression<Func<T, T2>> propertySelector,
        Expression<Func<T2, bool>> propertyPredicate)
    {
        var memberExpression = propertySelector.Body as MemberExpression;

        if (memberExpression == null)
        {
            throw new ArgumentException("propertySelector");
        }

        var expr = Expression.Lambda<Func<T, bool>>(propertyPredicate.Body, propertySelector.Parameters);
        var rebinder = new ParameterToMemberExpressionRebinder(propertyPredicate.Parameters[0], memberExpression);
        expr = (Expression<Func<T, bool>>)rebinder.Visit(expr);

        return expr;
    }

    class OrderLine
    {
    }

    class Order
    {
        public List<OrderLine> Lines;
    }

    static void test()
    {
        Expression<Func<Order, List<OrderLine>>> selectOrderLines = o => o.Lines;
        Expression<Func<List<OrderLine>, Boolean>> validateOrderLines = lines => lines.Count > 0;
        var validateOrder = ParameterToMemberExpressionRebinder.CombinePropertySelectorWithPredicate(selectOrderLines, validateOrderLines);

        // validateOrder: {o => (o.Lines.Count > 0)}
    }
}

Ben,你还在吗?我需要进行一些特殊的合并操作,你可能想要帮我解决这个问题。我试图将你的代码更改为我的需求,但没有成功。https://stackoverflow.com/questions/55440167/how-to-merge-object-expressions?noredirect=1#comment97597118_55440167 - Sturla

3

这个扩展功能如下:

public static class Utility
    {
        public static Expression<T> Compose<T>(this Expression<T> first, Expression<T> second, Func<Expression, Expression, Expression> merge)
        {
            // build parameter map (from parameters of second to parameters of first)
            var map = first.Parameters.Select((f, i) => new { f, s = second.Parameters[i] }).ToDictionary(p => p.s, p => p.f);

            // replace parameters in the second lambda expression with parameters from the first
            var secondBody = ParameterRebinder.ReplaceParameters(map, second.Body);

            // apply composition of lambda expression bodies to parameters from the first expression 
            return Expression.Lambda<T>(merge(first.Body, secondBody), first.Parameters);
        }

        public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> first, Expression<Func<T, bool>> second)
        {
            return first.Compose(second, Expression.And);
        }

        public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> first, Expression<Func<T, bool>> second)
        {
            return first.Compose(second, Expression.Or);
        }
    }

使用示例:

Expression<Func<Product, bool>> filter1 = p => a.ProductId == 1;
Expression<Func<Product, bool>> filter2 = p => a.Text.StartWith("test");
Expression<Func<Product, bool>> filterCombined = filter1.And(filter2);

我需要做什么才能在这个代码中添加括号呢? 我的意思是,如果我想要构建一个过滤器,例如(ProductId ==1 or ProductId ==2) and a.text.StartsWith("a")。 - Marty
1
我在哪里可以找到parameterRebinder参数重新绑定器? - h8red
2
如果出于某种原因您仍在寻找parameterRebinder,可以在MSDN博客上找到它。 - Igoris Azanovas

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接