我需要构建一个系统,其中我有许多存储在文件中的表达式。这些表达式将被读入程序,编译成 linq 表达式并作为函数对许多对象进行评估。但是,这些表达式将包含对代码中某些函数的引用(即它们不会仅由基本的 C# 运算符组成)。
我正在尝试使用 System.Linq.Dynamic 中的 DynamicExpression,它几乎可以工作,除了无法识别我的代码函数。基本上,当我有以下代码时:
public class GeoObj
{
public int layer;
public Polygon poly;
}
class Program
{
public static bool ComplicatedMethod(GeoObj x, GeoObj y)
{
bool result = // some quite complicated computation on x and y, say a polygon intersection test
return result;
}
static void Main(string[] args)
{
GeoObj o1 = new GeoObj(); // here set o1 fields
GeoObj o2 = new GeoObj(); // here set o2 fields
string sExpr = @"(x.layer == y.layer) && (ComplicatedMethod(x,y))";
var xparam = Expression.Parameter(typeof(GeoObj), "x");
var yparam = Expression.Parameter(typeof(GeoObj), "y");
Expression<Func<GeoObj, GeoObj, bool>> expr = (Expression<Func<GeoObj, GeoObj, bool>>)System.Linq.Dynamic.DynamicExpression.ParseLambda(new[] { xparam, yparam }, typeof(bool), sExpr);
var del2 = expr.Compile();
Console.WriteLine(del2(o1, o2));
}
}
(当然,在最终版本中,sExpr 将从文本文件中读取,但这只是作为示例。)
代码在 ParseLambda 处抛出异常,指出“ComplicatedFunction”未知。我可以看出 ParseLambda 如何不知道“ComplicatedFunction”,因为它不知道它在哪个程序集中使用。
有没有办法将该字符串编译成实际的表达式?如果我使用像这样的硬编码在我的代码中创建表达式:
Expression<Func<GeoObj, GeoObj, bool>> expr = (x, y) => (x.layer == y.layer) && ComplicatedFunction(x,y));
它按我的预期工作,但我无法预料所有可能的表达式。
如果不可能,我想下一个最好的办法是将字符串表达式解析为解析树,然后从该树创建表达式。由于只有几个函数(如“ComplicatedFunction”)将被使用,所以它看起来是可行的。从这样的表达式创建语法树的最佳方法/最佳代码是什么?我需要一些在符号未知时不会抛出异常的东西(例如“ComplicatedFunction”),并且可以让我轻松解析树以构建表达式。
谢谢。
-------------------- 更新------------------------ ------
@pil0t 的回答让我走上了解决这个问题的正确轨道。基本上,您必须下载 Dynamic Linq 的原始源文件并进行两次修改。
寻找:
static readonly Type[] predefinedTypes = {
并删除 'readonly' 关键字(以便可以对其进行修改)。
然后,将以下函数添加到 ExpressionParser 类中:
public static void AddPredefinedType(Type type)
{
predefinedTypes = predefinedTypes.Union(new[] { type }).ToArray();
keywords = ExpressionParser.CreateKeywords();
}
您可以像这样使用新功能:
using System.Linq.Dynamic;
...
ExpressionParser.AddPredefinedType(typeof(GeoObj));
关于表达式语法的注意事项。新添加的函数将使您能够访问“GeoObj”类(或您使用的任何类)的方法,并且只能访问这些方法。这意味着您必须按如下方式修改表达式:
class GeoObj {
public bool field1;
public bool method1() { /* return something */ }
public static ComplicatedMethod(GeoObj o1, GeoObj o2) { ... }
}
...
string myExpr = @"x.field1 && y.method1() && GeoObj.ComplicatedMethod(x,y)";
换句话说,您需要使用的所有函数都需要移到类中,就像@Shlomo 暗示的那样。我会喜欢一个更清晰的符号(例如使用“ComplicatedMethod(x,y)”而不是“GeoObj.ComplicatedMethod(x,y)”,或者使用类似 Method1(x)而不是 x.Method1()),但是在这一点上,这些主要是审美偏好。此外,由于可以使用的此类方法的数量很少,因此可以在内部重写表达式,从类似于函数调用的表达式变为方法调用。
感谢大家给我指明了正确的方向。
最佳答案
我通过修补原始资源来做到这一点
public class Foo
{
...
public static void AddPredefinedType(Type type)
{
ExpressionParser.predefinedTypes = ExpressionParser.predefinedTypes.Union(new[] { type }).ToArray();
ExpressionParser.CreateKeywords();
}
}
之后,你可以使用
Foo.AddPredefinedType(typeof(Program));
并且程序中的所有方法都将可用
关于c# - System.Linq.Dynamic.DynamicExpression 用方法解析表达式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31653697/