内容简述
-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软件包的描述。
具体的描述参间下面的程序中的注释:
1 package com.fjn.jdk.jvm.options.javaagent; 2 3 import java.io.IOException; 4 import java.lang.instrument.Instrumentation; 5 6 import com.sun.tools.attach.AttachNotSupportedException; 7 8 public class CustomAgent { 9 /** 10 * 如果Agent是通过JVM选项的方式捆绑到程序中,则在JVM初化完毕后,会执行premain方法,premain执行之后才是程序的main方法。 11 * 清单文件中需要指定Premain-Class
12 * <p> 13 * premain有两种形式,默认会执行1), 如果没有1)则会执行2), 1)和2)只会执行一个<br> 14 * <code> 15 * 1) public static void premain(String agentArgs, Instrumentation instrumentation)<br/> 16 * 2) public static void premain(String agentArgs) 17 * </code></p> 18 * 19 * @param agentArgs 20 * @param instrumentation 21 */ 22 public static void premain(String agentArgs, Instrumentation instrumentation) { 23 System.out 24 .println("CustomAgent#premain(String agentArgs, Instrumentation instrumentation)"); 25 parseAgentArgs(agentArgs); 26 } 27 28 public static void premain(String agentArgs) { 29 System.out.println("CustomAgent#premain(String agentArgs)"); 30 parseAgentArgs(agentArgs); 31 } 32 32 33 /** 34 * 如果Agent是在程序运行过程中,动态的捆绑到程序中,则是执行agentmain方法。 35 * 清单文件中要指定 Agent-Class 36 * <p> 37 * agentmain有两种形式,默认会执行1), 如果没有1)则会执行2), 1)和2)只会执行一个<br> 38 * <code> 39 * 1) public static void agentmain(String agentArgs, Instrumentation instrumentation)<br/> 40 * 2) public static void agentmain(String agentArgs) 41 * </code></p> 42 * 43 * 通过程序捆绑的代码:<br/> 44 * <code> 45 * VirtualMachine vm=VirtualMachine.attach("PID"); //给指定的进程捆绑agent<br/> 46 * 在得到目标进程的vm后,就可以通过 47 * vm.loadAgent("agentjar"),vm.loadAgentLibrary(dll), and loadAgentPath(dllPath) 进行捆绑操作了 <br/> 48 * 其中:<br> 49 * loadAgent是捆绑一个jar文件, 50 * loadAgentLibrary,loadAgentPath则是捆绑本地方法库(动态连接库) 51 * </code> 52 * 53 * @param agentArgs 54 * @param inst 55 */ 56 public static void agentmain(String agentArgs, Instrumentation inst) { 57 System.out 58 .println("CustomAgent#agentmain(String agentArgs, Instrumentation instrumentation)"); 59 parseAgentArgs(agentArgs); 60 } 61 62 public static void agentmain(String agentArgs) { 63 System.out.println("CustomAgent#agentmain(String agentArgs)"); 64 parseAgentArgs(agentArgs); 65 } 66 67 /** 68 * 不论是premain,还在agentmain,都可以指定参数,参数是一个字符串,具体怎么解析,是程序自己的事 69 * @param agentArgs 70 * @return 71 * @throws IOException 72 * @throws AttachNotSupportedException 73 */ 74 private static boolean parseAgentArgs(String agentArgs) { 75 boolean hasArgs = false; 76 if (agentArgs != null && !agentArgs.isEmpty()) { 77 System.out.println("agentArgs is : " + agentArgs); 78 hasArgs = true; 79 } else { 80 System.out.println("has no agentArgs ."); 81 } 82 83 return hasArgs; 84 } 85 }
将该类打包成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...
想要了解动态捆绑机制,可以参考: