javassist在pre-main方法中加载一个类文件(java instrumentation)

Man*_*eer 14 java instrumentation javassist

我正在尝试使用javassist加载一个特定的类,我在pre-main方法中执行此操作,如下所示:

public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
                        ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException 
{
   byte[] byteCode = classfileBuffer;

   if(className.toLowerCase().endsWith("class1"))
   {
       ClassPool classPool = ClassPool.getDefault();
       CtClass ctClass = classPool.get("com.class2");
   }
}
Run Code Online (Sandbox Code Playgroud)

但不幸的是,我得到一个NotFoundException ....

com.class2是一个在加载class1后将由类加载器加载的类,但我发现为class1添加一个返回类型为com.class2的新方法

我尝试了stackoverflow.com内的所有解决方案,但没有任何结果.....

最后,class1和class2都在同一个包中并且在同一个JAR文件中,class1有一个类型为class2的成员,但我不知道为什么这个classpool无法加载第二个.

这里的堆栈跟踪:

Exception: javassist.NotFoundException: com.Class2
 javassist.NotFoundException: com.Class2
at javassist.ClassPool.get(ClassPool.java:439)
at javassist.ClassPool.getCtClass(ClassPool.java:504)
at com.stuff.MainAppletTransformer.transform(MainAppletTransformer.java:69)
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 sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader.defineClassHelper(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader.access$100(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader$2.run(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader$2.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at sun.plugin2.applet.Plugin2ClassLoader.findClassHelper(Unknown Source)
at sun.plugin2.applet.Applet2ClassLoader.findClass(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader.loadClass0(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader.loadClass(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.plugin2.applet.Plugin2ClassLoader.loadCode(Unknown Source)
at sun.plugin2.applet.Plugin2Manager.initAppletAdapter(Unknown Source)
at sun.plugin2.applet.Plugin2Manager$AppletExecutionRunnable.run(Unknown Source)
at java.lang.Thread.run(Unknown Source)
Run Code Online (Sandbox Code Playgroud)

Raf*_*ter 13

您的方法没有返回值,因此无法编译.但我仍然可以告诉你出了什么问题:你没有设置正确的搜索路径.从您的堆栈跟踪中,您似乎正在运行一个受特定屏蔽的applet ClassLoader.您当前版本中的Javassist不会看到此类加载器.

通过使用ClassPool.getDefault(),您正在查找系统类路径中的信息.对于您的applet,您可能需要添加由transformatio方法移交的类加载器作为参数.

您可以将类加载器附加到Javassist的搜索路径

ClassPool classPool = ClassPool.getDefault();
classPool.appendClassPath(new LoaderClassPath(classLoader));
Run Code Online (Sandbox Code Playgroud)

添加此路径后,您的示例应该有效.