不同于于我们之前使用-Djava.library.path 指定其它的非Java类包用于实现JNI
javaagent本身就是命令的一个选项,用于Java类在加载前,做一些不可告人的事情cuiyaonan2000@163.com,
这里所说的不可告人的事情其实就是:字节码插桩
字节码插桩 听起来似乎很高大上的样子,实际上就是在Java应用启动加载类前通过某种机制在执行的字节码实现嵌入代码片段或者修改字节码的操作。有时候我们想要在执行 jar 程序前添加一些策略机制或者对某个执行方法进行修改操作,那么就可以通过 字节码插桩 技术实现
参考资料:
Javaagent是java命令的一个参数。参数 javaagent 可以用于指定一个 jar 包,并且对该 java 包有2个要求:
premain 方法,从字面上理解,就是运行在 main 函数之前的的类。当Java 虚拟机启动时,在执行 main 函数之前,JVM 会先运行-javaagent
所指定 jar 包内 Premain-Class 这个类的 premain 方法 。
在如上的截图中
-agentlib:
[=<选项>] 加载本机代理库
, 例如 -agentlib:hprof 另请参阅 -agentlib:jdwp=help 和 -agentlib:hprof=help -agentpath:
[=<选项>] 按完整路径名加载本机代理库
-javaagent:
[=<选项>] 加载 Java 编程语言代理, 请参阅 java.lang.instrument
在上面-javaagent
参数中提到了参阅java.lang.instrument
,这是在rt.jar
中定义的一个包,该路径下有两个重要的接口类:
该包提供了一些工具帮助开发人员在 Java 程序运行时,动态修改系统中的 Class 类型。其中,使用该软件包的一个关键组件就是 Javaagent。从名字上看,似乎是个 Java 代理类,而实际上,他的功能更像是一个Class 类型的转换器,他可以在运行时接受重新外部请求,对Class类型进行修改。-------换句话说就是类加载的时候将原字节码的类转换成另外的类,然后再把类加载到JVM cuiyaonan2000@163.com
从本质上讲,Java Agent 是一个遵循一组严格约定的常规 Java 类。 上面说到 javaagent命令要求指定的类中必须要有premain()方法,并且对premain方法的签名也有要求,签名必须满足以下两种格式:
- public static void premain(String agentArgs, Instrumentation inst)
-
- public static void premain(String agentArgs)
JVM 会优先加载 带 Instrumentation
签名的方法,加载成功忽略第二种,如果第一种没有,则加载第二种方法。这个逻辑在sun.instrument.InstrumentationImpl 类中:
如上最重要的2个接口:
ClassFileTransformer: 通过字面意思就是类文件转换器,具体转换的实现接口类
Instrumentation: 这个其实就是管理ClassFileTransformer 和出发ClassFileTransformer的一个管理类。
故此Instrumentation是盘活javaagent的主要类,它的主要方法如下所示;
- public interface Instrumentation {
-
- //增加一个Class 文件的转换器,转换器用于改变 Class 二进制流的数据,参数 canRetransform 设置是否允许重新转换。
- void addTransformer(ClassFileTransformer transformer, boolean canRetransform);
-
- //在类加载之前,重新定义 Class 文件,ClassDefinition 表示对一个类新的定义,如果在类加载之后,需要使用 retransformClasses 方法重新定义。addTransformer方法配置之后,后续的类加载都会被Transformer拦截。对于已经加载过的类,可以执行retransformClasses来重新触发这个Transformer的拦截。类加载的字节码被修改后,除非再次被retransform,否则不会恢复。
- void addTransformer(ClassFileTransformer transformer);
-
- //删除一个类转换器
- boolean removeTransformer(ClassFileTransformer transformer);
-
- boolean isRetransformClassesSupported();
-
- //在类加载之后,重新定义 Class。这个很重要,该方法是1.6 之后加入的,事实上,该方法是 update 了一个类。
- void retransformClasses(Class>... classes) throws UnmodifiableClassException;
-
- boolean isRedefineClassesSupported();
-
-
- void redefineClasses(ClassDefinition... definitions)
- throws ClassNotFoundException, UnmodifiableClassException;
-
- boolean isModifiableClass(Class> theClass);
-
- @SuppressWarnings("rawtypes")
- Class[] getAllLoadedClasses();
-
-
- @SuppressWarnings("rawtypes")
- Class[] getInitiatedClasses(ClassLoader loader);
-
- //获取一个对象的大小
- long getObjectSize(Object objectToSize);
-
-
-
- void appendToBootstrapClassLoaderSearch(JarFile jarfile);
-
-
- void appendToSystemClassLoaderSearch(JarFile jarfile);
-
-
- boolean isNativeMethodPrefixSupported();
-
-
- void setNativeMethodPrefix(ClassFileTransformer transformer, String prefix);
- }
使用 javaagent 需要几个步骤:
在执行以上步骤后,JVM 会先执行 premain 方法,大部分类加载都会通过该方法,而不是全部类。没有经过该方法的主要是系统类,因为很多系统类先于 agent 执行,而用户类的加载肯定是会被拦截的(换句话说就是我们自己的类肯定会经过该方法)。既然可以拦截类的加载,那么就可以去做重写类这样的操作,结合第三方的字节码编译工具,比如ASM,javassist,cglib等等来改写实现类。 ----这里很重要,比如skywalking的无侵入式监控就是使用agent的做了拦截处理cuiyaonan2000@163.com
如上所说,创建javaagen包含manifest.mf文件用于告诉程序执行的入口。
我们如果直接使用maven命令打包的话,工程会为我们自动生成一个manifest.mf文件。因此我们就可以使用maven插件来帮助我们生成想要的
- <plugin>
- <groupId>org.apache.maven.pluginsgroupId>
- <artifactId>maven-jar-pluginartifactId>
- <version>3.1.0version>
- <configuration>
- <archive>
-
- <manifest>
- <addClasspath>trueaddClasspath>
- manifest>
- <manifestEntries>
- <Premain-Class>com.rickiyang.learn.PreMainTraceAgentPremain-Class>
- <Agent-Class>com.rickiyang.learn.PreMainTraceAgentAgent-Class>
- <Can-Redefine-Classes>trueCan-Redefine-Classes>
- <Can-Retransform-Classes>trueCan-Retransform-Classes>
- manifestEntries>
- archive>
- configuration>
- plugin>
关于manifest.mf的选项说明如下所示:
上面介绍的Instrumentation是在 JDK 1.5中提供的,开发者只能在main加载之前添加手脚,在 Java SE 6 的 Instrumentation 当中,提供了一个新的代理操作方法:agentmain,可以在 main 函数开始运行之后再运行。
具体参考javaagent使用指南 - rickiyang - 博客园