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”时,会出现相同的错误。这个问题已经涵盖了这一点,但我并不真正理解那里的解决方案。
有这等后用同样的问题,但没有答案。
请帮忙。谢谢
好的,这就是我解决这个问题的方法。我忽略了一些错误,也学到了一些东西。我分享这个是因为我在任何地方都找不到直接解决问题的内容。参考这个答案和这篇文章:
的insertBefore,insertAfter和类似的方法, 不能访问该方法的任何局部变量,但可以访问方法的参数由他们的名字。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 我会在遇到它们时添加更多提示,欢迎任何更正。