java - 使用 Javassist 实现 ClassFileTransformer

标签 java instrumentation javassist

不同的来源提供了使用 Javassist 实现 ClassFileTransformer 的不同方法:

blog.newrelic.com/2014/09/29/diving-bytecode-manipulation-creating-audit-log-asm-javassist/

public byte[] transform(...) {
    ClassPool pool = ClassPool.getDefault(); // Edited to simplify
    pool.insertClassPath(new ByteArrayClassPath(className, classfileBuffer));
    CtClass cclass = pool.get(className.replaceAll("/", "."));
    ...
    return cclass.toBytecode();
}
<小时/>

blog.javabenchmark.org/2013/05/java-instrumentation-tutorial.html

public byte[] transform(...) {
    ClassPool cp = ClassPool.getDefault();
    CtClass cc = cp.get("org.javabenchmark.instrumentation.Sleeping");
    ...
    return cc.to:byteCode(); // edited to simplify
}
<小时/>

http://javapapers.com/core-java/java-instrumentation/

public byte[] transform(...) {
    ClassPool classPool = ClassPool.getDefault();
    CtClass ctClass = classPool.makeClass(new ByteArrayInputStream(classfileBuffer));
    ...
    return ctClass.to:byteCode(); // edited to simplify
}
<小时/>

哪个是最正确的方法,为什么?还有其他解决方案比这三个更好吗?

https://stackoverflow.com/a/26725215/776884提到设置正确的类加载器。使用instrumentation API时需要吗? ClassPool classPool = new ClassPool()CtClass.makeClass() 是否应该与仪器 API 一起使用?

最佳答案

所有示例都是错误的,并且在一般设置中不起作用。您永远不应该使用默认的类池,并且永远不应该(如 new relic 博客中所示)在转换器之间共享类池,因为您无法判断加载的类是否与其类加载器相关。

考虑一个应用程序服务器,其中每个应用程序都有自己的类加载器;您甚至无法使用默认类池(它引用系统类加载器,即类路径)看到转换后的类,并且您不能保证应用程序服务器上的所有应用程序都包含某个类,并且它们都包含它们。

唯一正确的解决方案是:

ClassPool classPool = new ClassPool();
classPool.appendClassPath(new LoaderClassPath(loader));
CtClass ctClass = classPool.makeClass(new ByteArrayInputStream(classfileBuffer));
这样,您就可以考虑到每个类都是由不同的类加载器加载的,这些类加载器可以表示不同的 View ,即包含与类路径不同的类或类的不同版本,并且您仍然可以解析提供的字节数组,该字节数组可以包含由其他 Java 代理先前触发的转换器添加的新成员。

关于java - 使用 Javassist 实现 ClassFileTransformer,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35975762/

相关文章:

c++函数地址在附加的探查器库中与主题代码库中不同

Java新手: How to use Java Agent From the Command Line for Instrumentation

java - 使用@JsonAnySetter 映射jackson 返回带有javassist 类的Unrecognized 字段

java - 停止先前的 Handler postdelayed 并开始新的

java - 从 JFileChooser 获取 file.properties 并在 ResourceBundle 上使用它

java - "VerifyError: Expecting to find object/array on stack"在Java中使用ASM监控对象创建时?

java - 如何使用 javaagent 访问工具方法的变量?

java - 使用 EL 传递 `<portlet:param>` 中的值

java - .class 文件的名称和位置在哪里定义?

java - 带有 Javassist 3.18.2-GA 的 Powermock 1.6.3 产生 NoSuchMethodError