JavaSe:-javaagent,-agentlib,-agentpath

时间:2022-03-24 05:44:46

内容简述

 

-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...

想要了解动态捆绑机制,可以参考:

http://ifeve.com/jvm-attach/