首页 > 编程语言 >Java代理之Java Agent分析

Java代理之Java Agent分析

时间:2024-11-08 18:32:14浏览次数:1  
标签:Java 字节 1.3 代理 Agent 修改

目录

1 Java Agent

1.1 简介

1.1.1 定义

Java Agent 是一种用于在 Java 应用启动或运行过程中对其进行监控、修改或增强的机制。它利用 Java Instrumentation API,可以在应用启动时或运行中动态加载代码,对目标应用的字节码进行操作。这种机制非常适合应用监控、性能分析、调试、代码注入和安全性增强等任务。

简单来说,Java Agent 就是运行在 Java 虚拟机(JVM)上的一种工具,能在程序运行时对其进行监控、修改甚至重定义。它的作用和 AOP(面向切面编程)有点类似,但更加底层,直接作用在 JVM 层面。可以理解为它是全局的 AOP,能在类加载、方法执行等时刻动态插手程序行为。

1.1.2 与代理区别

Java Agent 可以说是一种“代理”工具,但它的代理作用和一般的代理(例如 Java 中的 Proxy 类)有些不同。Java Agent 主要是通过修改类字节码的方式,实现在不直接修改原始代码的情况下对程序的运行行为进行增强或拦截。

Java Agent 和普通代理的区别:

  • 字节码层面的代理:Java Agent 是在类加载时,通过字节码操作来修改类的定义,因此属于低层次的代理。这不同于使用 Java 动态代理或 CGLIB 代理,它不需要在代码中显式调用代理方法。
  • 无侵入性:Java Agent 能够在应用启动或运行时注入代理逻辑,不需要修改原始代码。比如 APM 工具的 Java Agent 就能自动为应用添加性能监控,无需在每个方法中手动添加监控代码。
  • 全局作用:Java Agent 可以对 JVM 中的所有类进行代理操作(包括 JDK 自带类),并不是针对某个对象或接口的代理。代理逻辑可以应用于整个 JVM 中加载的所有类,适用范围更广。

与普通代理的对比

特性 Java Agent Java 动态代理 / CGLIB 代理
代理方式 字节码操作 接口或子类方法拦截
实现时机 JVM 启动时 / 运行时注入 编码时指定代理逻辑
侵入性 无侵入,自动加载 需要在代码中显式调用代理类
作用范围 全局所有类 某个对象或接口
典型用途 性能监控、日志注入、调试等 业务逻辑中的代理模式

1.1.3 主要功能和用途

主要作用:

  • 性能监控:可以捕获应用程序的性能数据,比如方法调用次数、执行时间、内存消耗等,生成性能报告。例如,常见的 APM(应用性能监控)工具如 New Relic、Dynatrace 等都使用了 Java Agent 技术。
  • 字节码增强:在类加载时修改类的字节码,比如添加日志、修改方法逻辑、实现代码注入等。Java Agent 可以在应用运行时拦截并修改方法,使其在不改变原始代码的情况下增加额外功能。
  • 动态调试:在不重启应用的情况下动态附加 Java Agent,可以实时监控或调试生产环境中的问题。
  • 应用安全性:可以为应用增加安全性检查,例如在方法调用前加入权限验证,或在检测到异常行为时触发报警。
  • 测试增强:可以利用 Java Agent 对应用内部行为进行模拟或监控,增强自动化测试或集成测试的功能。

1.2 原理和模式

Java Agent 使用 java.lang.instrument.Instrumentation 接口来对类的字节码进行修改。其基本流程如下:

  • 创建代理类:编写一个含有 premainagentmain 方法的代理类。premain 用于在应用启动时加载,agentmain 用于在应用运行时动态附加。
  • 实现字节码操作:在代理类中,通过 Instrumentation 对象,可以拦截和修改字节码,比如用 Java ASM 或 Javassist 等字节码工具来修改类文件。
  • 打包和运行:将代理类打包为 jar 并设置清单文件中的 Premain-Class 或 Agent-Class 属性,使 Java 在启动时加载该代理。

Java Agent 主要有两种模式:Premain模式和Agentmain模式:

  • Premain模式:在程序启动前就能注入
    这种模式通常是我们在程序启动时就注入 Agent,常见于应用启动时的初始化操作。
    例如:在程序启动时,配置一些监控、日志、性能分析工具。通过这种方式,Agent 可以在应用的生命周期中,从一开始就进行干预。
    使用场景:初始化操作、性能监控、日志收集等。
  • Agentmain模式:动态注入Agent
    这种模式是指在程序启动后,动态地将 Agent 注入到正在运行的 JVM 中。在主程序已经启动并且运行的过程中,也可以通过一些工具(比如 attach API)把 Agent 加入到 JVM 中。
    这种方式主要用于热更新和动态调试。
    使用场景:热部署、动态调整配置、动态监控等。

1.3 使用实现

1.3.1 Premain 模式

1.3.1.1 创建Agent类

首先,我们需要创建一个 Java 类,通常这个类会有一个静态方法 premain,它会在主程序启动前被执行。

import java.lang.instrument.Instrumentation;

public class MyAgent {
    // premain方法会在main方法之前执行
    public static void premain(String agentArgs, Instrumentation inst) {
        System.out.println("Java Agent initialized!");
        // 注册一个类的转换器
        inst.addTransformer(new MyClassFileTransformer());
    }
	static class MyClassFileTransformer implements ClassFileTransformer {
        @Override
        public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
                                ProtectionDomain protectionDomain, byte[] classfileBuffer) {
            // 这里可以对字节码进行修改
            System.out.println("Transforming class: " + className);
            return classfileBuffer; // 返回修改后的字节码
        }
    }
}

1.3.1.2 配置Maven

我们需要通过 Maven 配置项目的构建方式,将这个 Agent 类打包成一个 JAR 文件。关键在于 MANIFEST.MF 文件中的配置,需要指定 Agent 类的入口点。

<build>
  <plugins>
    <plugin>
      <groupId>org.apache.maven.plugins</groupId>
      <artifactId>maven-jar-plugin</artifactId>
      <version>3.1.0</version>
      <configuration>
        <archive>
          <manifestEntries>
            <Premain-Class>com.example.MyAgent</Premain-Class>
          </manifestEntries>
        </archive>
      </configuration>
    </plugin>
  </plugins>
</build>

1.3.1.3 启动程序时指定

打包好之后,我们只需在启动程序时通过 -javaagent 参数来指定这个 Agent JAR 文件。例如:

java -javaagent:/path/to/myagent.jar -jar myapp.jar

参数说明:

  • -javaagent:参数后面跟的是一个 Java 代理的 JAR 文件路径。这个代理可以在应用程序启动之前或运行期间对字节码进行修改或增强,常用于性能监控、日志记录等功能。
  • /path/to/myagent.jarJava 代理的 JAR 文件的完整路径
  • -jar myapp.jar:指定了要运行的主应用程序的 JAR 文件路径

1.3.2 Agentmain模式

假如要在程序运行时动态注入一个 Java Agent,可以使用 Agentmain 模式。这种方式可以在程序启动之后,通过附加到一个已经在运行的 JVM 来注入代码。

1.3.2.1 通过 Attach API 动态注入

这种方式依赖于 Attach API,它允许在程序运行时,将一个新的 Agent 附加到正在运行的 JVM 上。

import com.sun.tools.attach.*;

public class AgentAttacher {
    public static void main(String[] args) throws Exception {
        String pid = args[0];  // 获取目标进程的PID
        String agentJarPath = args[1];  // 要注入的Agent路径

        // 获取目标JVM的虚拟机进程
        VirtualMachine vm = VirtualMachine.attach(pid);
        // 向目标JVM进程注入Agent
        vm.loadAgent(agentJarPath);
        vm.detach();  // 注入后断开与目标JVM的连接
    }
}

这段代码通过 VirtualMachine.attach(pid) 连接到目标 JVM 进程,然后通过 loadAgent() 方法将 Java Agent 动态注入。这里的 pid 就是目标 JVM 进程的 ID,你可以通过工具(如 jps)来获取。

1.3.2.2 启动Agent

Premain模式 不同的是,Agent 在这种模式下并不需要在程序启动时就指定,而是可以在程序运行中后期动态地附加进去。

1.4 Instrumentation接口

InstrumentationJava Agent 的核心接口,它提供了修改和操作 JVM 中加载的类的能力。通过 Instrumentation,可以修改类字节码、重定义已有类,甚至能在类加载时插手,动态地修改类行为。

1.4.1 核心功能

Instrumentation 的核心功能:
Instrumentation 接口的功能非常丰富,以下是一些关键功能及其用途:

  • 修改类定义:
    可以在类加载前,通过 ClassFileTransformer 对类字节码进行修改。
    使用 redefineClasses 方法在类已经加载后重新定义该类,这样可以在运行时修改类的行为。
  • 添加和移除 ClassFileTransformer
    ClassFileTransformer 是一个用于修改类字节码的接口。通过 Instrumentation,可以将一个 ClassFileTransformer 添加到 JVM 中,监控或更改所有类的字节码。
    可以使用 addTransformer 方法将 ClassFileTransformer 添加到 Instrumentation 实例中,之后每次加载类时都会触发 transform 方法进行字节码修改。
  • 获取对象大小:
    使用 getObjectSize(Object object) 可以获取某个对象的大小,主要用于内存分析工具中。它可以精确地获取 Java 对象在内存中的占用空间。
  • 动态代理:
    Instrumentation 可以在运行时创建动态代理类,这样可以为现有的对象添加新的方法或行为。代理类可以拦截方法调用,实现方法增强。
  • 检索所有加载的类:
    getAllLoadedClasses() 方法可以返回 JVM 中所有已经加载的类,方便进行全局监控或分析。
  • 检测类是否已加载:
    使用 isModifiableClass(Class<?> theClass) 方法可以检查某个类是否可以修改,以避免对不支持的类进行重新定义而导致错误。
  • 添加类卸载事件处理器:
    Instrumentation 提供了类卸载的通知支持,可以用来监控类的卸载事件。可以用于记录对象的生命周期,监控资源的使用情况等。

1.4.2 典型用法

以下是一些 Instrumentation 的常见用法场景:

  • 性能监控工具(APM):
    可以通过 ClassFileTransformer 修改类字节码,添加方法进入和退出的时间记录,从而计算方法的执行时间,并汇总性能数据。
  • 内存监控:
    可以通过 getObjectSize 方法估算内存中对象的实际大小,结合类加载监控来分析内存泄露等问题。
  • 调试和测试工具:
    可以对类的行为进行修改,注入调试信息或测试代码。
    例如,在测试时可以通过 redefineClasses 修改类定义,不用重启应用来验证新代码的逻辑。
  • 安全增强:
    在类加载时对字节码进行检查或修改,防止某些不安全的操作或方法被调用,提高程序的安全性。

1.4.3 操作示例

以下是一个例子,展示了如何使用 Instrumentation 来修改类的字节码:

import java.lang.instrument.*;

public class MyClassTransformer implements ClassFileTransformer {
    @Override
    public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
                            ProtectionDomain protectionDomain, byte[] classfileBuffer) {
        if (className.equals("com/example/MyClass")) {
            // 这里可以使用 Javassist 或 ASM 等库来修改字节码
            System.out.println("Transforming MyClass...");
            // 返回修改后的字节码
            return modifiedClassBytecode;
        }
        return null;
    }
}

Java Agent 最常见的应用之一是性能监控。举个例子,我们可以通过 Agent 动态地修改类的字节码,来插入一些监控代码,记录方法执行时间、内存使用等信息。通过这种方式,我们无需修改现有代码,只需通过 Agent 即可实现监控。

比如,要监控某个方法的执行时间,可以在方法的入口和出口插入日志代码,记录执行时间:

public class MyClass {
    public void myMethod() {
        long start = System.currentTimeMillis();
        // 方法逻辑
        long end = System.currentTimeMillis();
        System.out.println("Method executed in " + (end - start) + " ms");
    }
}

通过 Agent 插入这个监控代码,可以动态获取到该方法的执行时间,无需修改源代码。

标签:Java,字节,1.3,代理,Agent,修改
From: https://www.cnblogs.com/jingzh/p/18535625

相关文章

  • ES6代理和反射新特性,详细讲解
    代理与反射es6新增了代理和反射特性,这两个特性为开发者提供了拦截并向基本操作嵌入额外行为的能力。代理基础<!DOCTYPEhtml><htmllang="en"><head><metacharset="UTF-8"><metacharset="UTF-8"><metahttp-equiv="X-UA-Compatib......
  • 【java类的生命周期】
    java类的生命周期大阶段加载>使用>卸载其中加载阶段分为加载>链接>初始化链接过程包含:验证>准备>解析加载阶段加载>链接(验证->准备->解析)>初始化加载将.class文件加载到jvm中,这个阶段,jvm根据类的全限定名称获取定义该类的二进制字节流,......
  • JAVA毕业设计198—基于Java+Springboot+vue3的健身房管理系统(源代码+数据库)
    毕设所有选题:https://blog.csdn.net/2303_76227485/article/details/131104075基于Java+Springboot+vue3的健身房管理系统(源代码+数据库)198一、系统介绍本项目前后端分离(可以改为ssm版本),分为用户、管理员两种角色1、用户:注册、登录、公告、论坛交流、健身课程购买......
  • Java流程控制-循环结构
    循环结构while循环while是最基本的循环,它的结构为:while(布尔表达式){ //循环内容}只要布尔表达式为true,循环就会一直执行下去。我们大多数情况是会让循环停止下来的,我们需要一个让表达式失效的方式来结束循环。少部分情况需要循环一直执行,比如服务器的请求响应监听等。......
  • Nginx反向代理之proxy_redirect指令【转】
    proxy_redirect该指令是用来重置头信息中的"Location"和"Refresh"的值。语法:proxy_redirectredirectreplacement;proxy_redirectdefault;proxy_redirectoff;默认值:proxy_redirectdefault;编写位置:它可以存储在http、server、location里面现在客户端向代理服务器......
  • Java+SpringBoot+Vue 学院个人信息管理系统
    学生个人信息管理一:基本介绍开发环境功能模块图系统功能部分数据库表设计用例分析二:部分系统页面展示登录页面首页管理端首页个人信息管理教师信息管理学生作业管理学习课程信息管理课程分类管理班级管理学生信息管理源码一:基本介绍开发环境·开发语言:Java·......
  • 如何从Java工程师成长为架构师?
    程序员进阶关于普通程序员的进阶之路,网上的学习路线与理论说辞一抓一大把。在小编看来优秀工程师的成长之路就是一条不断打怪升级之路,Java入行容易精通难,需要能力也需要运气。相信很多的Java程序员在自己的进阶之路上都有遇到以下几个方面的问题却不知道如何解决,也没有一个好......
  • 【Linux】为终端命令自定义快件键并弹窗提醒 设置快捷键切换网络代理(Network Proxy)Dis
    【Linux】为终端命令自定义快件键并弹窗提醒设置快捷键切换网络代理(NetworkProxy)Disabled/Manual并弹窗提醒可以自定义快捷键执行终端命令,执行完毕会有弹窗提醒。下面给一个例子,设置快捷键切换网络代理(NetworkProxy)Disabled/Manual并弹窗提醒。适用于Ubuntu系统,为......
  • AI巨头下场!Computer use横空出世! 揭秘实在Agent和Claude能力有啥区别?|实在Agent研究
    从Claude发布了像人一样操作电脑的能力之后,一直有小伙伴问我,实在Agent和Claude的computeruse能力有什么异同点。废话不多说,我们直接测试一波。AI巨头下场!揭秘实在Agent和Claude能力有啥区别?第一步、安装。目前Claude的computeruse,需要通过代码来部署,且需要提前充值;而实......
  • 解锁Java编程新高度!用validate注解做校验,让你的代码更高效、更安全!
    在Java中,@Valid注解通常用于验证对象的属性。它通常与Spring框架一起使用,以自动触发对JavaBean的验证。以下是如何使用@Valid注解进行校验的详细步骤和示例代码:1.添加依赖首先,确保你的项目中包含了SpringBoot的starter-web依赖,因为我们需要用到Spring的验证功能。<depend......