内容简述
-javaagent、-agentlib、-agentpath
-agentlib:libname[=options]
用于装载本地lib包;
其中libname为本地代理库文件名,默认搜索路径为环境变量PATH中的路径,options为传给本地库启动时的参数,多个参数之间用逗号分隔。在Windows平台上jvm搜索本地库名为libname.dll的文件,在linux上jvm搜索本地库名为libname.so的文件,搜索路径环境变量在不同系统上有所不同,比如Solaries上就默认搜索LD_LIBRARY_PATH。
比如:-agentlib:hprof
用来获取jvm的运行情况,包括CPU、内存、线程等的运行数据,并可输出到指定文件中;windows中搜索路径为JRE_HOME/bin/hprof.dll。
-agentpath:pathname[=options]
按全路径装载本地库,不再搜索PATH中的路径;其他功能和agentlib相同;更多的信息待续,在后续的JVMTI部分会详述。
-javaagent:jarpath[=options]
指定jvm启动时装入java语言基础设施代理。jarpath文件中的mainfest文件必须有Premain-Class(启动前捆绑时需要), Agent-Class(运行时捆绑时需要)属性。代理类也必须实现公共的静态public static void premain(String agentArgs, Instrumentation inst)方法(和main方法类似)。当jvm初始化时,将按代理类的说明顺序调用premain方法;具体参见java.lang.instrument软件包的描述。
具体的描述参间下面的程序中的注释:
-javaagent 示例
package com.fjn.jdk.jvm.options.javaagent; import java.io.IOException; import java.lang.instrument.Instrumentation; import com.sun.tools.attach.AttachNotSupportedException; public class CustomAgent { /** * 如果Agent是通过JVM选项的方式捆绑到程序中,则在JVM初化完毕后,会执行premain方法,premain执行之后才是程序的main方法。 * 清单文件中需要指定Premain-Class 12 * <p> * premain有两种形式,默认会执行1), 如果没有1)则会执行2), 1)和2)只会执行一个<br> * <code> * 1) public static void premain(String agentArgs, Instrumentation instrumentation)<br/> * 2) public static void premain(String agentArgs) * </code></p> * * @param agentArgs * @param instrumentation */ public static void premain(String agentArgs, Instrumentation instrumentation) { System.out .println("CustomAgent#premain(String agentArgs, Instrumentation instrumentation)"); parseAgentArgs(agentArgs); } public static void premain(String agentArgs) { System.out.println("CustomAgent#premain(String agentArgs)"); parseAgentArgs(agentArgs); } /** * 如果Agent是在程序运行过程中,动态的捆绑到程序中,则是执行agentmain方法。 * 清单文件中要指定 Agent-Class * <p> * agentmain有两种形式,默认会执行1), 如果没有1)则会执行2), 1)和2)只会执行一个<br> * <code> * 1) public static void agentmain(String agentArgs, Instrumentation instrumentation)<br/> * 2) public static void agentmain(String agentArgs) * </code></p> * * 通过程序捆绑的代码:<br/> * <code> * VirtualMachine vm=VirtualMachine.attach("PID"); //给指定的进程捆绑agent<br/> * 在得到目标进程的vm后,就可以通过 * vm.loadAgent("agentjar"),vm.loadAgentLibrary(dll), and loadAgentPath(dllPath) 进行捆绑操作了 <br/> * 其中:<br> * loadAgent是捆绑一个jar文件, * loadAgentLibrary,loadAgentPath则是捆绑本地方法库(动态连接库) * </code> * * @param agentArgs * @param inst */ public static void agentmain(String agentArgs, Instrumentation inst) { System.out .println("CustomAgent#agentmain(String agentArgs, Instrumentation instrumentation)"); parseAgentArgs(agentArgs); } public static void agentmain(String agentArgs) { System.out.println("CustomAgent#agentmain(String agentArgs)"); parseAgentArgs(agentArgs); } /** * 不论是premain,还在agentmain,都可以指定参数,参数是一个字符串,具体怎么解析,是程序自己的事 * @param agentArgs * @return * @throws IOException * @throws AttachNotSupportedException */ private static boolean parseAgentArgs(String agentArgs) { boolean hasArgs = false; if (agentArgs != null && !agentArgs.isEmpty()) { System.out.println("agentArgs is : " + agentArgs); hasArgs = true; } else { System.out.println("has no agentArgs ."); } return hasArgs; } }
将该类打包成customagent.jar,放到D盘根目录下,
清单文件:
Manifest-Version: 1.0 Premain-Class: com.fjn.jdk.jvm.options.javaagent.CustomAgent Sealed: true
测试代码:
package com.fjn.jdk.jvm.options.javaagent; import org.junit.Test; public class JavaAgentTest { /** * 在启动时,使用-javaagent方式加入代理 * <code> * -javaagent:D:/customagent.jar="Here, your can input agent arguments" * 如果要指定参数值 * </code> */ @Test public void testPremain(){ System.out.println("test premain, sepcial inst..."); } }
执行结果:
CustomAgent#premain(String agentArgs, Instrumentation instrumentation) agentArgs is : Here, your can input agent arguments test premain, sepcial inst...
想要了解动态捆绑机制,可以参考: