是否可以解释 C# 表达式树以发出 JavaScript?

Posted

技术标签:

【中文标题】是否可以解释 C# 表达式树以发出 JavaScript?【英文标题】:Is it possible to interpret a C# expression tree to emit JavaScript? 【发布时间】:2011-08-25 08:21:11 【问题描述】:

例如,如果您有这样的表达式:

Expression<Func<int, int>> fn = x => x * x;

有什么东西会遍历表达式树并生成它吗?

"function(x)  return x * x; "

【问题讨论】:

【参考方案1】:

这可能并不容易,但是是的,这是绝对可行的。 Entity Framework 或 Linq to SQL 等 ORM 将 Linq 查询转换为 SQL,但实际上您可以从表达式树中生成任何您想要的内容...

您应该实现ExpressionVisitor 来分析和转换表达式。


编辑:这是一个适用于您的示例的非常基本的实现:

Expression<Func<int, int>> fn = x => x * x;
var visitor = new JsExpressionVisitor();
visitor.Visit(fn);
Console.WriteLine(visitor.javascriptCode);

...

class JsExpressionVisitor : ExpressionVisitor

    private readonly StringBuilder _builder;

    public JsExpressionVisitor()
    
        _builder = new StringBuilder();
    

    public string JavaScriptCode
    
        get  return _builder.ToString(); 
    

    public override Expression Visit(Expression node)
    
        _builder.Clear();
        return base.Visit(node);
    

    protected override Expression VisitParameter(ParameterExpression node)
    
        _builder.Append(node.Name);
        base.VisitParameter(node);
        return node;
    

    protected override Expression VisitBinary(BinaryExpression node)
    
        base.Visit(node.Left);
        _builder.Append(GetOperator(node.NodeType));
        base.Visit(node.Right);
        return node;
    

    protected override Expression VisitLambda<T>(Expression<T> node)
    
        _builder.Append("function(");
        for (int i = 0; i < node.Parameters.Count; i++)
        
            if (i > 0)
                _builder.Append(", ");
            _builder.Append(node.Parameters[i].Name);
        
        _builder.Append(") ");
        if (node.Body.Type != typeof(void))
        
            _builder.Append("return ");
        
        base.Visit(node.Body);
        _builder.Append("; ");
        return node;
    

    private static string GetOperator(ExpressionType nodeType)
    
        switch (nodeType)
        
            case ExpressionType.Add:
                return " + ";
            case ExpressionType.Multiply:
                return " * ";
            case ExpressionType.Subtract:
                return " - ";
            case ExpressionType.Divide:
                return " / ";
            case ExpressionType.Assign:
                return " = ";
            case ExpressionType.Equal:
                return " == ";
            case ExpressionType.NotEqual:
                return " != ";

            // TODO: Add other operators...
        
        throw new NotImplementedException("Operator not implemented");
    

它只处理单个指令的 lambda,但无论如何 C# 编译器无法为块 lambda 生成表达式树。

当然还有很多工作要做,这是一个非常简单的实现...您可能需要添加方法调用 (VisitMethodCall)、属性和字段访问 (VisitMember) 等。

【讨论】:

【参考方案2】:

C# 编译器已经为您解析表达式;剩下的就是遍历表达式树并生成代码。遍历树可以递归完成,每个节点可以通过检查它的类型来处理(Expression 有几个子类,代表例如函数、运算符和成员查找)。每种类型的处理程序都可以生成适当的代码并遍历节点的子节点(根据它是哪种表达式类型,这些子节点将在不同的属性中可用)。例如,可以通过首先输出“function(”后跟参数名称后跟“)”来处理函数节点。然后,可以递归处理正文,最后输出“”。

【讨论】:

谢谢,'traverse' 和 'generate' 是更准确的动词,我已经更新了问题【参考方案3】:

Script# 被 Microsoft 内部开发人员用于执行此操作。

【讨论】:

+0...实际上 Script# 是相当完整的交叉编译器...它确实转换了 C# 源代码并且不能轻易地用于转换 Expression 对象。 我的理解是,OP 将 Expression 对象用作是否可以将 C# 代码编译为 JavaScript 的理论示例,而不是将它们用作实际实现。【参考方案4】:

一些人开发了开源库来解决这个问题。我一直在看的是Linq2CodeDom,它将表达式转换为CodeDom图,然后只要代码兼容,就可以编译成JavaScript。

Script# 利用原始 C# 源代码和编译的程序集,而不是表达式树。

我对 Linq2CodeDom 做了一些小的编辑,以将 JScript 添加为受支持的语言——本质上只是添加对 Microsoft.JScript 的引用,更新枚举,并在 GenerateCode 中添加一个新的案例。下面是转换表达式的代码:

var c = new CodeDomGenerator();
c.AddNamespace("Example")
    .AddClass("Container")
    .AddMethod(
        MemberAttributes.Public | MemberAttributes.Static,
        (int x) => "Square",
        Emit.@return<int, int>(x => x * x)
    );
Console.WriteLine(c.GenerateCode(CodeDomGenerator.Language.JScript));

结果如下:

package Example

    public class Container
    
        public static function Square(x : int)
        
            return (x * x);
        
    

方法签名反映了 JScript 的强类型特性。最好使用 Linq2CodeDom 生成 C#,然后将其传递给 Script# 以将其转换为 JavaScript。我相信第一个答案是最正确的,但是正如您通过查看 Linq2CodeDom 源代码所看到的,在处理每个案例以正确生成代码方面需要付出很多努力。

【讨论】:

【参考方案5】:

看看Lambda2Js,这是一个由Miguel Angelo为此目的创建的库。

它将CompileToJavascript 扩展方法添加到任何表达式。

示例 1:

Expression<Func<MyClass, object>> expr = x => x.PhonesByName["Miguel"].DDD == 32 | x.Phones.Length != 1;
var js = expr.CompileToJavascript();
Assert.AreEqual("PhonesByName[\"Miguel\"].DDD==32|Phones.length!=1", js);

示例 2:

Expression<Func<MyClass, object>> expr = x => x.Phones.FirstOrDefault(p => p.DDD > 10);
var js = expr.CompileToJavascript();
Assert.AreEqual("System.Linq.Enumerable.FirstOrDefault(Phones,function(p)return p.DDD>10;)", js);

更多示例here。

【讨论】:

以上是关于是否可以解释 C# 表达式树以发出 JavaScript?的主要内容,如果未能解决你的问题,请参考以下文章

构建动态表达式树以过滤集合属性

如何创建 LINQ 表达式树以选择匿名类型

如何创建表达式树以执行与“StartsWith”相同的操作

改变谓词的表达式树以针对另一种类型

展开 extjs 树以查看一个元素

C#中的多态