classPool.get(className)throws RuntimeException找不到类

时间:2017-09-04 04:49:22

标签: bytecode java-bytecode-asm javassist bytecode-manipulation javaagents

我正在尝试使用Javassist编写一个简单的检测代理。

public class Agent implements ClassFileTransformer {

    protected Instrumentation instrumentation;
    protected ClassPool classPool;

    public Agent(Instrumentation instrumentation){

        this.instrumentation = instrumentation;
        this.classPool = ClassPool.getDefault();
        this.instrumentation.addTransformer(this);
    }

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

        try {

            System.out.printf("%s.transform: %s\n", this.getClass().getCanonicalName(), className);

            classPool.insertClassPath(new ByteArrayClassPath(className, classfileBuffer));

            CtClass ctClass = classPool.get(className);  /* <- throws error */
        }
        catch (Exception ex){
            System.out.println(ex);
            return null;
        }
    }
}

但我收到以下错误:

  

java.lang.RuntimeException:找不到java / lang / invoke / MethodHandleImpl:java / lang / invoke / MethodHandleImpl.class中的java.lang.invoke.MethodHandleImpl

唯一没有抛出此异常的类是IDE项目(IntelliJ IDEA)中的那个。

我尝试手动添加类路径中的路径,但这也无济于事。即。

String[] cpEntries = System.getProperty("java.class.path").split(File.pathSeparator);
for (String cpEntry : classpathEntries)
    classPool.insertClassPath(cpEntry);

我错过了什么?谢谢!

1 个答案:

答案 0 :(得分:1)

问题是className的分隔符是/,而classPool.get(className)需要.分隔符,因此修复是替换这些字符,即:

String classNameDotted = className.replace('/', '.');
CtClass ctClass = classPool.get(classNameDotted);