如何在没有调用的情况下合并两个 C# Lambda 表达式?

Posted

技术标签:

【中文标题】如何在没有调用的情况下合并两个 C# Lambda 表达式?【英文标题】:How to merge two C# Lambda Expressions without an invoke? 【发布时间】:2010-11-11 00:55:40 【问题描述】:

我想合并以下表达式:

// 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;

我该怎么做?

【问题讨论】:

【参考方案1】:

最优雅的方式是使用Expression Visitor。特别是,MSDN Blog Entry 描述了如何使用它来组合谓词(使用布尔 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 你还在做任何改变吗?我需要做一些特殊的合并,你可能想帮我解决这个问题吗? ***.com/questions/55440167/… 试图将你的更改为我的需要 github.com/sturlath/CombineObjectExpressionsTest/blob/… 没有运气【参考方案2】:

此扩展有效:

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) 和 a.text.StartsWith("a") 在哪里可以找到 parameterRebinder? 如果由于某种原因你还在寻找 parameterRebinder,可以在MSDN blog找到它

以上是关于如何在没有调用的情况下合并两个 C# Lambda 表达式?的主要内容,如果未能解决你的问题,请参考以下文章

如何在没有公共密钥的情况下合并 Apache Spark 中的两个数据帧?

如何在不丢失 SQL 中的任何行的情况下合并两个表?

c#如何合并多个dll文件

如何在没有实际合并的情况下测试合并

如何在 C# 中合并两个数据表(没有任何公共标识符)?8

如何在不使用 Invoke 方法的情况下组合两个 lambda 表达式?