Lambda表达式在运行时C#中的多个条件

Rya*_*yan 7 c# lambda

我想知道如何通过输入多个参数来创建表达式树

例:

dataContext.Users.Where(u => u.username == "Username" && u.password == "Password")
Run Code Online (Sandbox Code Playgroud)

目前我所做的代码如下,但是想要更一般地考虑条件是OR还是AND

public Func<TLinqEntity, bool> ANDOnlyParams(string[] paramNames, object[] values)
    {
        List<ParameterExpression> paramList = new List<ParameterExpression>();
        foreach (string param in paramNames)
        {
            paramList.Add(Expression.Parameter(typeof(TLinqEntity), param));
        }

        List<LambdaExpression> lexList = new List<LambdaExpression>();
        for (int i = 0; i < paramNames.Length; i++)
        {
            if (i == 0)
            {
                Expression bodyInner = Expression.Equal(
                                    Expression.Property(
                                        paramList[i], paramNames[i]),
                                        Expression.Constant(values[i]));
                lexList.Add(Expression.Lambda(bodyInner, paramList[i]));
            }
            else
            {
                Expression bodyOuter = Expression.And(
                                    Expression.Equal(
                                    Expression.Property(
                                    paramList[i], paramNames[i]),
                                    Expression.Constant(values[i])),
                                    Expression.Invoke(lexList[i - 1], paramList[i]));
                lexList.Add(Expression.Lambda(bodyOuter, paramList[i]));
            }
        }

        return ((Expression<Func<TLinqEntity, bool>>)lexList[lexList.Count - 1]).Compile();
    }
Run Code Online (Sandbox Code Playgroud)

谢谢

Dan*_*ker 2

Expression.And这里使用的是错误的东西,它是按位与。你要AndAlso

除此之外,您似乎已经知道如何构建表达式树的机制。所以你真正要问的是如何让构建方法的调用者指定一种更复杂、更灵活的方式来组合不同的条件。

最终为了真正的灵活性,您需要一种迷你查询语言。解析语言以构建表达式树。

在短期内,您可能会使用更简单的东西:原始表达式列表和 bool 标志来说明它们是否应该与 && 或 || 组合。

更新- 我注意到您实际上正在将结果表达式编译成真正的委托。这让我想知道为什么你一开始要这么艰难地做这件事。为什么不像您最初的示例那样将表达式写为 lambda?(如果您使用的是 Linq to SQL 或 EF,则无论如何都不应该编译该表达式。)

更新 2 - 您可能需要的是Dynamic Linq