1

这与我关于拦截对给定类中的字段的所有访问的问题有关,而不仅仅是那些以与 Groovy“属性”样式访问一致的方式完成的访问。您可以在此处查看:在 groovy 中拦截 LOCAL 属性访问

我发现肯定会解决我的问题的一种方法是在编译时使用 AST 用属性访问重写任何非属性访问。例如,如果一个类看起来像这样:

class Foo {
  def x = 1
  def getter() {
    x
  }
  def getProperty(String name) {
    this."$name" ++
  }
}

foo = new Foo()
assert foo.getter() == 1
assert foo.x == 2

这些断言语句将起作用,因为 getter 方法直接访问 x 并且 foo.x 通过 getProperty("x") 在返回之前递增 x。

经过一些试验和错误后,我可以使用 AST 转换来更改代码的行为,以便“getter”方法中的表达式“x”实际上作为属性而不是本地字段进行访问。到目前为止,一切都很好!

现在,我该如何访问给定类中本地字段的所有访问权限?我一直在互联网上寻找某种 AST tree walker helper,但还没有找到。我真的需要在这里http://groovy.codehaus.org/api/org/codehaus/groovy/ast/expr/package-summary.html和所有 18 种语句类型http://groovy.codehaus.org/api/org/codehaus/groovy/ast/expr/package-summary.html实现一个表达式步行器: //groovy.codehaus.org/api/org/codehaus/groovy/ast/stmt/package-summary.html ? 这似乎是某人必须已经写过的东西(因为它首先是构建 AST 树的组成部分),但我似乎找不到它。

4

1 回答 1

2

格伦

您正在寻找某种访客。Groovy 定义了一些您可以使用的访问者(文档薄弱)。对于您的问题,我没有确切的答案,但我可以为您提供一些指导。

下面的代码片段显示了如何遍历类的 AST 并打印所有方法名称:

class TypeSystemUsageVisitor extends ClassCodeVisitorSupport {

@Override
public void visitExpression(MethodNode node) {
    super.visitMethod(node)
    println node.name
}

@Override
protected SourceUnit getSourceUnit() {
    // I don't know ho I should implement this, but it makes no difference
    return null;
} 
}

这就是我使用上面定义的访问者的方式

def visitor = new TypeSystemUsageVisitor()
def sourceFile = new File("path/to/Class.groovy")
def ast = new AstBuilder().buildFromString(CompilePhase.CONVERSION, false, sourceFile.text).find { it.class == ClassNode.class }
ast.visitContents(visitor)

游客会为您照顾横穿这棵树。他们有 visit* 方法,您可以覆盖这些方法并使用它们做任何您想做的事情。我相信适合您问题的访问者是 CodeVisitorSupport,它有一个 visitVariableExpression 方法。

我建议您阅读groovyConsole 附带的 AST 浏览器的代码,以获取有关如何使用 Groovy AST 访问者的更多示例。另外,请查看 CodeVisitorSupport 的api 文档

于 2012-12-09T22:40:58.617 回答