java agent简介
-
主要就是两种,一种的方法是premain,一种是agentmain。这两种的区别是:
- premain是在jvm启动的时候类加载到虚拟机之前执行的
- agentmain是可以在jvm启动后类已经加载到jvm中了,才去转换类。这种方式会转换会有一些限制,比如不能增加或移除字段。
-
具体的做法,两者的实际做法是差不多的:
-
premain
定义个静态方法
public static void premain(String args, Instrumentation inst)
,
在生成jar包中MANIFEST.MF文件中需要有
Premain-Class: xxx.xxx
,xxx.xxx就是上面premain方法所在的类名在java 的启动参数中添加 -javaagent:/jar包路径[=agentArgs]
这样定义了后jvm启动时,就会去加载javaagent中指定的jar包,查找MANIFEST.MF文件中Premain-Class属性的类,执行premain方法。
参考asm文档简单修改了下打印方法执行时间的demo
blogdemo/javabasedemo/agentdemo at main · wbo112/blogdemo (github.com)
-
agentmain
定义个静态方法
public static void agentmain(String agentOps, Instrumentation instrumentation)
,在生成jar包中MANIFEST.MF文件中需要有Agent-Class: xxx.xxx (xxx.xxx就是上面agentmain方法所在的类名)
Can-Retransform-Classes: true使用下面代码,将agent添加到指定java进程
vm = VirtualMachine.attach(pid); try { vm.loadAgent("D:\\tmp\\my-java-agent-1.0-jar-with-dependencies.jar", null); } finally { vm.detach(); }
这个github找到一个很不错的案例。wujiuye/bytecode-book: 《Java虚拟机字节码从入门到实战》一书的配套代码 (github.com)
-
-
具体类的转换处理一般都是用asm之类修改字节码的开源组件。主要就是实现ClassFileTransformer接口,对入参的byte[]这个就是class类的字节数组了,对这个进行转换,返回新的class类的byte[]字节数组