javassist.CannotCompileException:[source error]没有这样的字段:$ 1

时间:2017-03-02 09:08:47

标签: java code-injection bytecode instrumentation javassist

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

method.insertBefore("{ System.out.println($1);}");

此声明给出了以下错误:

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

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

this其他帖子有相同的问题,但没有回答。

请帮忙。谢谢

1 个答案:

答案 0 :(得分:3)

好吧,这就是我解决这个问题的方法。我忽略了很少的错误以及我学到的一些东西。我正在分享这个,因为我无法在任何地方找到直接解决问题的内容。 参考this回答和this文章:

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)错误
  • insertBeforeinsertAfter,需要单个语句或单个代码块,如上面第一行代码段所示。
  • 如果在no such field中使用变量,则会出现此错误ctmethod.addcatch()
  • ctmethod.addLocalVariable()有助于声明insertBefore()等可以访问的变量。但是,如果您尝试使用声明的变量,.addCatch()仍会出现no such field错误.addLocalVariable()
  • 由于我的愚蠢错误,我得到no such field: $1。我正在迭代一个类中的一些方法,所以错误是由于其中一些方法有零参数$1,$2.. and $args工作正常,如this doc。
  • 中所述

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