5

私は groovy の使用方法 (および拡張方法) を学ぼうとしています。このページの例に従っています。基本的に、Groovy コードのアノテーションを定義して、コンパイラ プロセスにフックする方法を示します。この例は、メソッド呼び出しの前後に行を出力する書き込みと注釈を中心に展開しています。

私のコードは次のようになります。最初に必要なインポート:

package foo

import org.codehaus.groovy.transform.*
import java.lang.annotation.*
import org.codehaus.groovy.ast.*
import org.codehaus.groovy.control.*
import org.codehaus.groovy.ast.stmt.*
import org.codehaus.groovy.ast.expr.*

次に、使用する注釈を定義します。

@Retention(RetentionPolicy.SOURCE)
@Target([ElementType.METHOD])
@GroovyASTTransformationClass(["foo.LoggingASTTransformation"])
public @interface WithLogging {
}

次に、変換自体:

@GroovyASTTransformation(phase=CompilePhase.SEMANTIC_ANALYSIS)
public class LoggingASTTransformation implements ASTTransformation {

  public void visit(ASTNode[] nodes, SourceUnit sourceUnit) {
      println("visiting astnodes")
      List methods = sourceUnit.getAST()?.getMethods()
      // find all methods annotated with @WithLogging
      methods.findAll { MethodNode method ->
          method.getAnnotations(new ClassNode(WithLogging))
      }.each { MethodNode method ->
          Statement startMessage = createPrintlnAst("Starting $method.name")
          Statement endMessage = createPrintlnAst("Ending $method.name")

          List existingStatements = method.getCode().getStatements()
          existingStatements.add(0, startMessage)
          existingStatements.add(endMessage)
      }
  }

  private Statement createPrintlnAst(String message) {
      return new ExpressionStatement(
          new MethodCallExpression(
              new VariableExpression("this"),
              new ConstantExpression("println"),
              new ArgumentListExpression(
                  new ConstantExpression(message)
              )
          )
      )
  }
}

最後に、この変換を使用するはずの私のコード:

public class Foo {
   @WithLogging
   def f() { println "hello from f" }
}

f = new Foo()
f.f()

さて、これはStarting f\n hello from f \n Ending fを出力することになっていますが、表示されるのはhello from fだけです(そこに私の問題があります)。コードからわかるように、変換自体に訪問中の astnodesメッセージを入れて、うまくいけばそこに到達するかどうかを確認しましたが、残念ながらそうではありません (またはそう思われます)。

groovy -version版画Groovy Version: 1.6.0 JVM: 1.6.0_11

誰でもこのコードを試して、自分のシステムで動作するかどうかを確認できますか、または何が間違っている可能性があるかについての指針を教えてください。

4

2 に答える 2

0

変換を使用すると思われるコードを変更してみてください

public class Foo
{
   @foo.WithLogging
   def f() { println "hello from f"}
....

私は同じ例を試し、ここで私の経験を共有しましたhttp://kartik-shah.blogspot.com/2009/03/groovy-16-ast-transformation-example_5323.html

于 2009-03-26T21:34:54.967 に答える