Groovy编译时AST转换:分配给字段

时间:2014-07-31 07:54:06

标签: groovy abstract-syntax-tree

我目前正在尝试实现一些Groovy编译时AST转换,但我遇到了麻烦:

如何为字段的赋值语句指定AST转换?即AST转换应转换以下代码:

class MyClass {
    @MyTransformation
    String myField

    public void init() {
    }
}

类似

class MyClass {
    String myField

    public void init() {
        this.myField = "initialized!"
    }
}

我尝试使用此AST构建器调用:

def ast = new AstBuilder().buildFromSpec {
        expression{
            declaration {
                variable "myField"
                token "="
                constant "initialized!"
            }
        }
}

但是在声明类的“init”方法中插入结果语句后,它会插入一个变量赋值,如

java.lang.Object myField = "initialized!"

我查看了Ast Builder TestCase中包含的示例,但它们仅涵盖了类主体中的字段声明,而不是字段的赋值。我自己尝试使用fieldNode导致编译器错误。我将编译阶段设置为INSTRUCTION_SELECTION;我认为这应该没问题。

我如何实现这一目标?基于AstBuilder#buildFromSpec方法的解决方案是首选,但任何帮助都将受到高度赞赏。

1 个答案:

答案 0 :(得分:5)

我通常建议不要使用AST构建器。它适用于原型设计,但您并不能真正控制其生成的内容。特别是,在这里,它无法处理您创建的变量表达式应引用字段节点的事实。 AST Builder非常了解AST,但不应该用于生产代码恕我直言。

这是一个自包含的示例,演示了如何实现您想要的效果。 @ASTTest中的代码对应于您的转换代码:

import groovy.transform.ASTTest                                                         
import org.codehaus.groovy.ast.expr.BinaryExpression                            
import org.codehaus.groovy.ast.expr.VariableExpression                          
import org.codehaus.groovy.ast.expr.ConstantExpression                          
import org.codehaus.groovy.ast.stmt.ExpressionStatement                         
import org.codehaus.groovy.syntax.Token                                         
import org.codehaus.groovy.syntax.Types      

class MyClass {                                                                         
    String myField                                                                      

    @ASTTest(phase=SEMANTIC_ANALYSIS,value={                                            
            def classNode = node.declaringClass                                         
            def field = classNode.getDeclaredField('myField')                           
            def assignment = new BinaryExpression(                                      
              new VariableExpression(field),                                              
              Token.newSymbol(Types.EQUAL, 0, 0),                                         
              new ConstantExpression('initialized!')                                      
            )                                                                                   
            node.code.addStatement(new ExpressionStatement(assignment))                 
    })                                                                          
    public void init() {                                                                
    }                                                                                   
}                                                                                       
def c = new MyClass()                                                                   
c.init()                                                                                
println c.myField

希望这有帮助!