如何使用括号动态创建“System.Linq.Expression”表达式

时间:2015-01-09 19:51:42

标签: c# linq dynamic expression

我需要动态创建表达式。我的问题是如何在System.Linq.Expression中添加括号来处理优先级......

int a = 1, b = 1, c = 1, d = 2;

我已将以下表达式转换为

 var v = a > 1 || b > 1 && c > 1 || d > 1;
 Expression case1Expr =

                Expression.AndAlso(
                Expression.OrElse(
                Expression.GreaterThan(Expression.Parameter(typeof(int), "a"), Expression.Constant(1)),
                Expression.GreaterThan(Expression.Parameter(typeof(int), "b"), Expression.Constant(1))),


                Expression.OrElse(
                Expression.GreaterThan(Expression.Parameter(typeof(int), "c"), Expression.Constant(1)),
                Expression.GreaterThan(Expression.Parameter(typeof(int), "d"), Expression.Constant(1))));

但如何用括号转换一个,

var v2 = (a > 1 || b > 1) && (c > 1 || d > 1);

4 个答案:

答案 0 :(得分:2)

只是为了扩展其他答案;括号是我们需要保持我们用来描述方程式(A OP B)的中缀符号的含义。其他符号,例如 postfix (A B OP),不需要它。

在你的情况下,你正在组合一个表达式树,这也是明确的,即只有一种方法来解释它。因此表达式树中没有圆括号的概念。树的“形状”决定了操作的顺序。

答案 1 :(得分:0)

您已经拥有这些语义的代码。当您使用确切的代码调用ToString时,您会得到:

(((a > 1) OrElse (b > 1)) AndAlso ((c > 1) OrElse (d > 1)))

不需要特定的表达式来表示括号;括号隐含在组成表达式的方式中。在您的代码中,您将Or表达式传递给And表达式,因此Or表达式将被括号括起来。

答案 2 :(得分:0)

您可以通过构造表达式树的方式指定优先级。 &&运算符通常优先于||运算符,因此您的转换将应用于带括号的表达式。否则它只会是:

var v = a > 1 || b > 1 && c > 1 || d > 1;

OrElse(
    GreaterThan(a > 1),

    OrElse(
      AndAlso(
        GreaterThan(b > 1),
        GreaterThan(c > 1)),
      GreaterThan(d > 1)))

答案 3 :(得分:-1)

我可能会错过你的问题,但你不会这样做:

Expression first = Expression.OrElse(
    Expression.GreaterThan(Expression.Parameter(typeof(int), "a"), Expression.Constant(1)),
    Expression.GreaterThan(Expression.Parameter(typeof(int), "b"), Expression.Constant(1)));

Expression second = Expression.OrElse(
    Expression.GreaterThan(Expression.Parameter(typeof(int), "c"), Expression.Constant(1)),
    Expression.GreaterThan(Expression.Parameter(typeof(int), "d"), Expression.Constant(1)));

Expression final = Expression.AndAlso(first, second);

结果:(((a > 1) OrElse (b > 1)) AndAlso ((c > 1) OrElse (d > 1)))