c# - System.Linq.Dynamic.DynamicExpression 用方法解析表达式

标签 c# linq expression-trees dynamic-linq

我需要构建一个系统,其中我有许多存储在文件中的表达式。这些表达式将被读入程序,编译成 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));

关于表达式语法的注意事项。新添加的函数将使您能够访问“Ge​​oObj”类(或您使用的任何类)的方法,并且只能访问这些方法。这意味着您必须按如下方式修改表达式:

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/

相关文章:

c# - 如何使用 Linq 连接两个表并计算一列匹配记录的行数

带有函数(x)的 VB.NET Linq 表达式未执行?

c# - 在 C# Lambda 表达式中使用 System.DateTime 会出现异常

c# - AuthenticationBuilder 不包含 AddAzureAd 的定义

C# List<T> 包含测试

c# - 为什么 Code First 类需要导航属性?

c# - .NET 的 Random() 的实现更改

c# - 如何在 JSON.Net 中将对象转换为字典?

c# - 在方法中使用 lambda 并匹配列表

entity-framework - 如何动态绑定(bind) Entity Framework 中替换表达式节点中的参数