5

我正在寻找一种将条件表达式解析为字符串的方法。

我能想到的最好的例子是 LINQ-to-SQL。它使用 ExpressionVisitor 来格式化“Where”子句。例子:

from a in b where a.x == 5 && a.y < 3 select a

这将转化为以下字符串(大约,MSSQL 对我来说不是最新的):

"SELECT * FROM b WHERE x = 5 AND y < 3"

根据我的阅读,这是使用 ExpressionVisitor 类完成的,如本文所述:链接

现在的问题是我不使用 LINQ,但我需要这个特殊的功能。有没有办法解析这样的条件?我愿意用反射、委托、lambda 等做任何事情。

老实说,我不认为这是可能的,但我的大脑有点炸了(阅读:如果这个问题很荒谬,那就太好了),所以我想我不妨试试 S/O。

编辑:最终用法示例:

// Usage:
foo.Bar(foo => foo.X == 5 && foo.Y < 3)

// Ideal string output (variable name (foo) is not needed):
"foo.X == 5 && foo.Y < 3"

编辑 2:是的,一个数字可以小于 3 等于 5。告诉你我的大脑被炸了。

4

2 回答 2

4

如果它是关于构建表达式树本身,那么您可以利用 C# 编译器功能。

只要知道 Func 的类型参数,将 lambda 表达式传递给函数接受 Expression> 是合法的。例如

 private static void PrintExpression(Expression<Func<int, bool>> lambda)
 {
      Console.WriteLine(lambda.ToString());
 }

可以称为

 PrintExpression(a=> a > 0 && a < 5);

您可以使用泛型即兴创作

private static void PrintExpression<T1,T2>(Expression<Func<T1, T2>> lambda)
{
      Console.WriteLine(lambda.ToString());
}

并用

   PrintExpression<int, bool>(a=> a > 0 && a < 5);

对于表达式部分的自定义打印,您可以编写一个简单的递归函数来打印表达式或任何其他适合您的逻辑。

请记住,lambda 表达式在编译时被编译成一个表达式 - 所以不能用已经编译的 Func 替换它。

作为替代方案,您始终可以构建自定义查询提供程序,但这会稍微偏离目的 - 因为您需要将其绑定为某种可查询的(再次自定义)。

于 2010-11-17T03:50:27.887 回答
3

尝试这样的事情:

static string GetExpressionString<T>(Expression<Func<T, bool>> expression)
{
    return expression.Body.ToString();
}

用法如下:

string s = GetExpressionString<Foo>(foo => foo.X == 5 && foo.Y < 3);

哪个会返回:

((foo.X = 5) && (foo.Y < 3))
于 2010-11-17T04:11:47.053 回答