主要就是两种,一种的方法是premain,一种是agentmain。这两种的区别是:
具体的做法,两者的实际做法是差不多的:
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[]字节数组