前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >动态追踪之java agent

动态追踪之java agent

作者头像
索码理
发布2022-12-28 14:53:48
8360
发布2022-12-28 14:53:48
举报
文章被收录于专栏:索码理

上篇文章我们说到阿里的诊断工具Arthas对方法和类的监控使用的是动态追踪技术,本文我们将介绍动态追踪技术Java Agent。

Java Agent是什么?

Java Agent技术,也被称为Java代理、Java探针,从JDK1.5它就出现了,它允许程序员利⽤其构建⼀个独⽴于应⽤程序的代理程序。Java Agent本身就是个jar包,它利用JVM提供的Instrumentation API来更改加载在JVM中的现有字节码,Java Agent可以理解为是JVM级别的AOP。

Java Agent 怎么用?

Java Agent使用包括两个部分:代码实现和配置文件。

  • 代码实现:这部分包括3个步骤:
  1. 实现ClassFileTransformer接口并重写transform方法。 ClassFileTransformer用于在JVM加载实现类之前转换类文件。
  2. 编写agent类 agent类中有两个方法:premain方法和agentmain方法。
    • premain用于在代理方法执行前调用,在 JVM 启动时必须为其指明Java代理。它有两个实现
    • agentmain用于在代理方法运行时执行。
  3. 修改配置文件并打包
  • 配置文件:配置文件名为MANIFEST.MF,需放在META-INF文件夹下或者在maven中配置。
代码语言:javascript
复制
Premain-Class:表示实现premain方法的类。
Agent-Class:表示实现agentmain方法的类。
Can-Redefine-Classes:表示Java agent是否可以重新定义被代理类。
Can-Retransform-Classes:表示是否允许Java Agent转换被代理类。

Java Agent的加载

Java Agent的加载分为静态加载和动态加载。

静态加载

在应用程序启动时加载Java代理称为静态加载,静态加载在任何代码执行之前在启动时修改字节码。它是以premain方法为入口的,premain方法有两个重载方法:

代码语言:javascript
复制
public static void premain(String agentArgs, Instrumentation inst)
public static void premain(String agentArgs)
  • agentArgs:字符串参数,通过-javaagent传递
  • inst:java.lang.instrument.Instrumentation类对象

JVM 会优先加载带 Instrumentation 对象参数的方法,加载成功忽略第二种;如果第一种没有,则加载第二种方法。

下面写个简单的例子玩一下静态加载:

  1. 新建一个maven项目并引入javassist包
代码语言:javascript
复制
 <dependency>
    <groupId>org.javassist</groupId>
    <artifactId>javassist</artifactId>
    <version>3.29.2-GA</version>
</dependency>
  1. 新建一个ClassFileTransformer接口实现类PrintMethodCostTransformer,用户打印方法耗时
代码语言:javascript
复制
public class PrintMethodCostTransformer implements ClassFileTransformer {

    private final String targetClassName;

    public PrintMethodCostTransformer(String targetClassName) {
        this.targetClassName = targetClassName;
    }

    @Override
    public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) {
        try {
            if (className != null) {
                className = className.replaceAll("/", ".");

                if (className.contains(targetClassName)) {
                    ClassPool classPool = ClassPool.getDefault();
                    classPool.insertClassPath(className);
                    CtClass ctClass = classPool.get(className);
                    CtMethod[] methods = ctClass.getDeclaredMethods();
                    if (methods != null && methods.length > 0) {
                        for (CtMethod ctMethod : methods) {
                            String methodName = ctMethod.getName();
                            if (!"main".equals(methodName)) {
                                //修改字节码,定义两个long类型变量
                                ctMethod.addLocalVariable("begin", CtClass.longType);
                                ctMethod.addLocalVariable("end", CtClass.longType);
                                //方法执行前操作
                                ctMethod.insertBefore("System.out.println(\"进入 ["+methodName+"] 方法\");");
                                ctMethod.insertBefore("begin = System.nanoTime();");
                                //方法执行后操作
                                ctMethod.insertAfter("end = System.nanoTime();");
                                ctMethod.insertAfter("System.out.println(\"方法 [" + methodName + "] 耗时:\"+ (end - begin) +\"ns\");");
                                ctMethod.insertAfter("System.out.println(\"退出 ["+methodName+"] 方法\");");
                            }
                        }
                        ctClass.detach();
                        return ctClass.toBytecode();
                    }
                }
            }
        } catch (Throwable e) {
            e.printStackTrace();
        }
        return new byte[0];
    }
}
  1. 定义一个agent代理类AgentClass
代码语言:javascript
复制
/**
 * @author 索码理(suncodernote)
 */
public class AgentClass {

    // main方法执行前的修改
    public static void premain(String agentArgs , Instrumentation inst){
        System.out.println("enter premain(String agentArgs , Instrumentation inst)");
        inst.addTransformer(new PrintMethodCostTransformer(agentArgs));
    }
}

修改配置文件并打包

4.1 在META-INF文件夹下文件夹下创建一个MANIFEST.MF文件

文件内容如下

代码语言:javascript
复制
Manifest-Version: 1.0
Created-By: 索码理(suncodernote)
Can-Redefine-Classes: true
Can-Retransform-Classes: true
Premain-Class: com.example.agent.AgentClass

4.2 在修改pom.xml,指定MANIFEST.MF文件覆盖掉自动生成的MANIFEST.MF文件。

代码语言:javascript
复制
<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-jar-plugin</artifactId>
    <configuration>
        <archive>
            <manifestFile>src/main/resources/META-INF/MANIFEST.MF</manifestFile>
        </archive>
    </configuration>
</plugin>

最后通过maven install打包成一个jar包,我这里打成的jar包完成路径为D:\Program Files\apache-maven-3.6.3\repository\BasicJava\basic_java_demos\1.0-SNAPSHOT\basic_java_demos-1.0-SNAPSHOT.jar。一个简单的静态加载就完成了,接下来进行测试。

静态加载测试

  1. 新建一个maven项目,并新建一个测试类

MainTest每隔两秒钟调用一次print方法

代码语言:javascript
复制
public class MainTest {

    public static void main(String[] args) throws InterruptedException {
        while (true){
            print("agents");
            TimeUnit.SECONDS.sleep(2);
        }
    }

    private static void print(String name){
        System.out.println("时间:"+LocalDateTime.now()+","+"hello "+name);
    }
}
  1. 在MainTest测试类添加启动参数
代码语言:javascript
复制
-javaagent:"D:\Program Files\apache-maven-3.6.3\repository\BasicJava\basic_java_demos\1.0-SNAPSHOT\basic_java_demos-1.0-SNAPSHOT.jar"=com.example.jvmlearing.agent.MainTest

上面这段代码是指定了上面打包的agent jar包的完成路径,并通过-javaagent命令向agent包传递参数com.example.jvmlearing.agent.MainTest

-Xbootclasspath/a:D:\javassist-3.28.0-GA.jar是引用javassist包,当然也可以通过maven引入到项目中

  1. 启动MainTest进行测试

控制台打印结果:

从结果中可以看到premain方法是最先启动的,而且打印了方法耗时。

动态加载

将Java代理加载到已经运行的JVM中的过程称为动态加载。它利用的是Java的Attach API,Attach API不是Java标准API,而是Sun公司提供的一套扩展API,用来向目标JVM”依附”(Attach)代理工具程序的。Attach 机制对外提供了一种进程间的通信能力,能让一个进程传递命令给 JVM;其次,Attach 机制内置一些重要功能,可供外部进程调用。比如 jstack 工具就是要依赖这个机制来工作的。

动态加载的入口是agentmain方法,同样它也有两个重载方法

代码语言:javascript
复制
public static void agentmain(String agentArgs, Instrumentation inst)
public static void agentmain(String agentArgs)

premain方法一样,JVM 会优先加载带 Instrumentation 对象参数的方法,加载成功忽略第二种;如果第一种没有,则加载第二种方法。

举个例子,简单玩一下动态加载:

  1. 在刚刚的java agent项目的AgentClass中加入agentmain方法
代码语言:javascript
复制
public class AgentClass {

    // main方法执行前的修改
    public static void premain(String agentArgs , Instrumentation inst){
        System.out.println("enter premain(String agentArgs , Instrumentation inst)");
        inst.addTransformer(new PrintMethodCostTransformer(agentArgs));
    }

    // 控制类运行时的行为
    public static void agentmain(String agentArgs , Instrumentation inst) throws UnmodifiableClassException, ClassNotFoundException {
        System.out.println("enter agentmain(String agentArgs , Instrumentation inst)");
        inst.addTransformer(new PrintMethodCostTransformer(agentArgs) , true);

        Class<?> claz = Class.forName(agentArgs);
        //要重新定义的类
        inst.retransformClasses(claz);
    }
}
  1. MANIFEST.MF文件中加入agent-class
  1. 重新再打包

动态加载测试

  1. 在刚刚测试的项目中引入本地依赖tools.jar
代码语言:javascript
复制
<dependency>
    <groupId>com.sun</groupId>
    <artifactId>tools</artifactId>
    <version>1.8.0</version>
    <scope>system</scope>
    <systemPath>${JAVA_HOME}\lib\tools.jar</systemPath>
</dependency>
  1. 新建一个测试类AttachTest
代码语言:javascript
复制
public class AttachTest {
    public static void main(String[] args) throws AgentLoadException, IOException, AgentInitializationException, AttachNotSupportedException, InterruptedException {
        //传递给java代理的参数
        String targetClassName = "com.example.jvmlearing.agent.MainTest";
        String jar = "D:\\Program Files\\apache-maven-3.6.3\\repository\\BasicJava\\basic_java_demos\\1.0-SNAPSHOT\\basic_java_demos-1.0-SNAPSHOT.jar";
        for (VirtualMachineDescriptor descriptor: VirtualMachine.list()) {
            String displayName = descriptor.displayName();
            if (displayName.equals(targetClassName)) {
                String processId = descriptor.id();
                System.out.println("process id="+processId);
                VirtualMachine virtualMachine = VirtualMachine.attach(processId);
                //加载代理jar
                virtualMachine.loadAgent(jar, targetClassName);
                //解除绑定
                virtualMachine.detach();
            }
        }
    }
}
  1. 删除测试类MainTest的启动参数并先后启动MainTest和AttachTest

可以看到agentmain方法是在MainTest运行时执行的。

启动AttachTest之后,可以看到AttachTest的控制台打印的MainTest的进程id

再通过jps命令看下MainTest的进程id

可以看到两个进程id是一样的,也说明动态加载依附的是虚拟机进程。

Attach API 很简单,只有2个主要的类,都在 com.sun.tools.attach 包里面:

  • VirtualMachine类:代表一个 Java 虚拟机,也就是程序需要监控的目标虚拟机,提供了获取系统信息(比如内存dump、线程dump,类信息统计)、加载代理程序、Attach 和 Detach 等方法 。
  • VirtualMachineDescriptor类:一个描述虚拟机的容器类,配合 VirtualMachine 类完成各种功能。

通过VirtualMachine类的attach(pid)方法,便可以attach到一个运行中的java进程上,之后便可以通loadAgent(agentJarPath)来将agent的jar包注入到对应的进程,然后对应的进程会调用agentmain方法。

静态加载和动态加载的区别

从上面的例子中可以发现静态加载是需要和被代理的程序一起启动,需要在启动的时候通过-javaagent参数指定静态加载的jar包,被代理的程序是“知道”自己被代理的。而动态加载则需要被代理程序先启动,只要获取到被代理程序的进程id,通过loadAgent方法指定动态加载jar包就行了,它属于热插拔式的。

总结

本篇文章我们分别使用Java Agent的静态加载和动态加载成功的对字节码进行了修改、追踪,并完成了一个打印方法耗时的简单示例。Java Agent能够访问加载到JVM中的类,它的应用十分广泛,可用于实现Java IDE的调试功能、热部署功能、线上诊断⼯具和性能分析⼯具。本篇只是触及了Java Agent的皮毛,感兴趣的可以深入了解一下。下篇文章将介绍一个动态追踪框架BTrace。

参考资料: https://www.baeldung.com/java-instrumentation https://www.developer.com/design/what-is-java-agent/ https://www.jianshu.com/p/6967d4dfbc49

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2022-10-12,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 索码理 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Java Agent是什么?
  • Java Agent 怎么用?
  • Java Agent的加载
    • 静态加载
      • 静态加载测试
    • 动态加载
      • 动态加载测试
    • 静态加载和动态加载的区别
    • 总结
    相关产品与服务
    容器服务
    腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
    领券
    问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档