javassist.CannotCompileException: [源错误] 没有这样的字段:$1

Gau*_*tam 5 java instrumentation bytecode code-injection javassist

参考:http : //jboss-javassist.github.io/javassist/tutorial/tutorial2.html

method.insertBefore("{ System.out.println($1);}");
Run Code Online (Sandbox Code Playgroud)

此语句给了我以下错误:

Exception: javassist.CannotCompileException: [source error] no such field: $1
javassist.CannotCompileException: [source error] no such field: $1
        at javassist.CtBehavior.insertBefore(CtBehavior.java:774)
        at javassist.CtBehavior.insertBefore(CtBehavior.java:734)
        at com.here.debugHelper.DurationTransformer.transform(DurationTransformer.java:124)
        at sun.instrument.TransformerManager.transform(Unknown Source)
        at sun.instrument.InstrumentationImpl.transform(Unknown Source)
        at java.lang.ClassLoader.defineClass1(Native Method)
        at java.lang.ClassLoader.defineClass(Unknown Source)
        at java.security.SecureClassLoader.defineClass(Unknown Source)
        at java.net.URLClassLoader.defineClass(Unknown Source)
        at java.net.URLClassLoader.access$100(Unknown Source)
        at java.net.URLClassLoader$1.run(Unknown Source)
        at java.net.URLClassLoader$1.run(Unknown Source)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at com.here.debugHelper.TestInstrumentation.main(TestInstrumentation.java:10)
Caused by: compile error: no such field: $1
        at javassist.compiler.TypeChecker.fieldAccess(TypeChecker.java:845)
        at javassist.compiler.TypeChecker.atFieldRead(TypeChecker.java:803)
        at javassist.compiler.TypeChecker.atMember(TypeChecker.java:988)
        at javassist.compiler.JvstTypeChecker.atMember(JvstTypeChecker.java:66)
        at javassist.compiler.ast.Member.accept(Member.java:39)
        at javassist.compiler.JvstTypeChecker.atMethodArgs(JvstTypeChecker.java:221)
        at javassist.compiler.TypeChecker.atMethodCallCore(TypeChecker.java:735)
        at javassist.compiler.TypeChecker.atCallExpr(TypeChecker.java:695)
        at   javassist.compiler.JvstTypeChecker.atCallExpr(JvstTypeChecker.java:157)
        at javassist.compiler.ast.CallExpr.accept(CallExpr.java:46)
        at javassist.compiler.CodeGen.doTypeCheck(CodeGen.java:242)
        at javassist.compiler.CodeGen.atStmnt(CodeGen.java:330)
        at javassist.compiler.ast.Stmnt.accept(Stmnt.java:50)
        at javassist.compiler.CodeGen.atStmnt(CodeGen.java:351)
        at javassist.compiler.ast.Stmnt.accept(Stmnt.java:50)
        at javassist.compiler.Javac.compileStmnt(Javac.java:567)
        at javassist.CtBehavior.insertBefore(CtBehavior.java:754)
        ... 17 more
Run Code Online (Sandbox Code Playgroud)

此外,当我使用任何参数(传递给参数)或任何局部变量而不是“$1”时,会出现相同的错误。这个问题已经涵盖了这一点,但我并不真正理解那里的解决方案。

等后用同样的问题,但没有答案。

请帮忙。谢谢

Gau*_*tam 6

好的,这就是我解决这个问题的方法。我忽略了一些错误,也学到了一些东西。我分享这个是因为我在任何地方都找不到直接解决问题的内容。参考这个答案和这篇文章:

insertBeforeinsertAfter类似的方法, 不能访问该方法的任何局部变量,但可以访问方法的参数由他们的名字。no such field如果尝试访问局部变量,则会发生错误。

作为检测过程,这些方法不能访问其中声明的任何参数。 ctmethod.insertBefore("{int i = 4; int j = 9;}") ctmethod.insertBefore("i = 5;"); ctMethod.insertAfter("j = 9;");

  • 上面的代码会no such field: j(and i)报错
  • insertBefore, insertAfter, 期望单个语句或单个代码块,如上述代码片段的第一行中所做的那样。
  • no such field如果也使用了变量,则会出现此错误ctmethod.addcatch()
  • ctmethod.addLocalVariable()有助于声明一个可由insertBefore()等访问的变量。但是,如果您尝试使用使用声明的变量,.addCatch()仍然会no such field出错.addLocalVariable()
  • no such field: $1由于我的愚蠢错误,我得到了。我正在迭代一个类中的许多方法,所以错误是由于其中一些方法的参数为零$1,$2.. and $args文档中所述,效果很好。

PS 我会在遇到它们时添加更多提示,欢迎任何更正。