javaagent两种加载方式及实例
所属分类 javaagent
浏览量 3047
两种获取Instrumentation接口实例的方法
agent on load premain
agent on attach agentmain
关注 JVMTI 事件 ClassFileLoadHook 事件
读取字节码文件之后回调 , 可对字节码进行重定义或重转换
premain 与 agentmain 的区别
premain和agentmain 都是为了回调Instrumentation实例并激活sun.instrument.InstrumentationImpl#transform()
从而回调注册到Instrumentation中的ClassFileTransformer实现字节码修改,功能基本一样。
两者主要区别
premain 是JDK1.5引入的,agentmain 是JDK1.6引入的
premain需要通过命令行使用外部代理jar包,即-javaagent:代理jar包路径;
agentmain 通过attach机制直接附着到目标VM中加载代理
premain 作用于JVM加载的所有类,所有类首次加载并且进入程序main()方法之前,premain方法会被激活,然后所有被加载的类都会执行ClassFileTransformer列表中的回调。
agentmain 采用attach机制,需要借助Instrumentation#retransformClasses(Class< ?>... classes)让对应的类重新转换,激活重新转换的类执行ClassFileTransformer列表中的回调。
premain 方式
public static void premain(String agentArgs, Instrumentation inst);
public static void premain(String agentArgs);
带Instrumentation参数的优先级高,会被优先调用
agentArgs 通过命令行参数传入
MANIFEST.MF 文件,必须包含 Premain-Class 选项,通常也会加入Can-Redefine-Classes 和 Can-Retransform-Classes 选项
使用参数 -javaagent: jar包路径启动代理
premain加载过程
1.创建并初始化 JPLISAgent
2.读取 MANIFEST.MF 文件参数,设置 JPLISAgent 参数
3.监听 VMInit 事件,在 JVM 初始化完成后
A 创建 InstrumentationImpl 对象
B 监听 ClassFileLoadHook 事件
C 调用 InstrumentationImpl 的loadClassAndCallPremain方法,在该方法里调用 javaagent 中 MANIFEST.MF 里指定的Premain-Class 类的 premain 方法
agentmain 方式
public static void agentmain(String agentArgs, Instrumentation inst);
public static void agentmain(String agentArgs);
// 列出所有VM实例
List list = VirtualMachine.list();
// attach目标VM
VirtualMachine.attach(descriptor.id());
// 目标VM加载Agent
VirtualMachine#loadAgent("代理Jar路径","命令参数");
agentmain 加载过程
1 创建并初始化JPLISAgent
2 解析MANIFEST.MF 里的参数 ,设置 JPLISAgent
3.监听 VMInit 事件,在 JVM 初始化完成之后
(1)创建 InstrumentationImpl 对象
(2)监听 ClassFileLoadHook 事件
(3)调用 InstrumentationImpl 的loadClassAndCallAgentmain方法,在该方法里调用javaagent里 MANIFEST.MF 里指定的Agent-Class类的agentmain方法。
Instrumentation 的一些限制
大多数情况下,使用Instrumentation都是使用其字节码插桩的功能
premain和agentmain两种方式修改字节码的时机都是类文件加载之后,必须要带有Class类型的参数,不能通过字节码文件和自定义的类名重新定义一个本来不存在的类。
类重定义Instrumentation#retransformClasses()方法,有以下限制
1.新类和老类的父类必须相同;
2.新类和老类实现的接口数也要相同,并且是相同的接口;
3.新类和老类访问符必须一致。 新类和老类字段数和字段名要一致;
4.新类和老类新增或删除的方法必须是private static/final修饰的;
5.可以删除修改方法体。
重新定义一全新类 可以创建新的类加载器去加载
MANIFEST.MF 文件 参数在 pom文件里指定
maven-jar-plugin manifestEntries
premain pom
https://gitee.com/dyyx/demos/blob/master/jvmtools/pom.xml
mainagent pom
https://gitee.com/dyyx/demos/blob/master/jvmtools/pom-main-agent.xml
premain agent 打包
mvn clean source:jar install -Dmaven.test.skip
main agent 打包
mvn clean source:jar install -Dmaven.test.skip -f pom-main-agent.xml
java -javaagent:jvmtools-1.0.jar -cp .:jvmtools-1.0.jar dyyx.agent.premain.PreMainTest
java -javaagent:jvmtools-1.0.jar=premain__args -cp .:jvmtools-1.0.jar dyyx.agent.premain.PreMainTest
java -cp .:jvmtools-1.0.jar dyyx.agent.ClientMain list
Caused by: java.lang.ClassNotFoundException: com.sun.tools.attach.VirtualMachine
java -cp .:jvmtools-1.0.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_121.jdk/Contents/Home/lib/tools.jar dyyx.agent.ClientMain list
java -cp .:jvmtools-1.0.jar:/Library/Java/JavaVirtualMachines/jdk1.8.0_121.jdk/Contents/Home/lib/tools.jar dyyx.agent.ClientMain attach 92108 /Users/dugang/tmp/jvmtools-1.0.jar
agent jar包需要使用绝对路径,否则会加载不到
Error opening zip file or JAR manifest missing: jvmtools-1.0.jar
两种agent例子代码
https://gitee.com/dyyx/demos/tree/master/jvmtools/src/main/java/dyyx/agent
上一篇
下一篇
银华日利(511880) vs 华宝添益(511990)
java debug 原理简介
javaagent机制
Java8日期api使用
jconsole连接jmx失败处理
大数据风控概述