揭秘Java Agent技术:解锁Java工具开发的新境界

简介: 作为JDK提供的关键机制,Java Agent技术不仅为Java工具的开发者提供了一个强大的框架,还为性能监控、故障诊断和动态代码修改等领域带来了革命性的变革。本文旨在全面解析Java Agent技术的应用场景以及实现方式,特别是静态加载模式和动态加载模式这两种关键模式。

一、Java Agent技术

Java Agent技术是JDK提供的关键机制,专门用于编写高级Java工具。通过这项技术,开发者能够创建特殊的JAR包,这些JAR包能够在Java程序运行时执行其中的代码。Java Agent技术赋予了Java程序执行独立Java Agent程序中代码的能力,这种执行方式分为静态加载模式和动态加载模式两种。

1.静态加载模式

静态加载模式允许在Java程序启动的初始阶段就执行指定的代码,因此特别适用于APM(应用性能管理)等需要从一开始就监控程序执行性能的场景。为实现静态加载,开发者需要在Java Agent项目中编写一个premain方法,并将其打包成JAR包。

public static void premain(string agentArgs, Instrumentation inst){
}

image.gif

之后,通过以下命令启动Java程序,Java虚拟机将自动加载并执行agent中的代码:

java -jar -javaagent:./agent.jar test.jar

image.gif

-javaagent: 告诉JVM在启动应用程序之前加载并应用这个Agent
./agent.jar Java Agent JAR文件的路径
test.jar 要运行的Java应用程序的JAR文件

premain方法将在主线程中执行。

image.gif

2.动态加载模式

与静态加载模式不同,动态加载模式允许在任意时刻执行Java Agent代码,因此特别适用于Arthas等诊断系统。实现动态加载,开发者需要编写一个agentmain方法,并将其打包成JAR包。

public static void agentmain(String agentArgs, Instrumentation inst){
}

image.gif

要执行Java Agent代码,开发者可以使用以下代码片段动态连接到指定进程ID的Java程序:

// 动态连接到指定进程ID的java程序
VirtualMachine vm= VirtualMachine.attach("进程ID");
// 加载iava agent
vm.loadAgent("./agent.jar");

image.gif

与premain方法不同,agentmain方法将在独立线程中执行。这种灵活性使得动态加载模式在需要动态修改或监控运行中的Java应用程序时特别有用。

image.gif

二、案例

1.搭建Java Agent静态加载模式的案例

步骤一:Maven项目构建与插件集成

首先,创建一个新的Maven项目,在项目的pom.xml文件中,添加maven-assembly-plugin插件的配置。这个插件的主要作用是将项目依赖和编译后的类文件打包成一个单独的JAR文件,该文件将作为Java Agent的部署包。

<plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-assembly-plugin</artifactId>
                <configuration>
                   <!-- 将所有依赖打入同一个jar包中 -->
                    <descriptorRefs>
                        <descriptorRef>jar-with-dependencies</descriptorRef>
                    </descriptorRefs>
                    <!--指定java agent相关配置文件-->
                    <archive>
                        <manifestFile>src/main/resources/MANIFEST.MF</manifestFile>
                    </archive>
                </configuration>
            </plugin>
image.gif

步骤二:编写Java Agent核心代码

在项目的源代码目录下,编写一个包含premain方法的类。premain方法是Java Agent的生命周期入口,它会在Java应用启动之前执行。在这个方法中,可以编写自定义的逻辑,例如打印一行信息。

public class AgentMain {
    // premain方法
    public static void premain(String agentArgs, Instrumentation inst){
        System.out.println("premain方法");
    }
}
image.gif

步骤三:定义Java Agent的清单文件

创建一个名为MANIFEST.MF的清单文件,并放置在项目的src/main/resources/META-INF目录下。这个文件描述了Java Agent的一些配置属性,比如使用哪个类的premain方法。

Manifest-Version: 1.0
Premain-Class: com.rye.javaagent.AgentMain
Can-Redefine-Classes: true
Can-Retransform-Classes: true
Can-Set-Native-Method-Prefix: true
image.gif
Manifest-Version: 1.0 指定了清单文件的版本,通常为1.0,它用于标识清单文件遵循的规范版本。
Premain-Class: com.rye.javaagent.AgentMain Premain-Class属性指定了Java Agent的主要入口点(包含premain方法的类)。当Java应用程序启动并加载该Agent时,premain方法会被调用。
Can-Redefine-Classes: true 这个属性允许Java Agent在运行时重新定义已加载的类。如果设置为true,Agent可以在不卸载和重新加载整个类的情况下修改类的字节码。这对于实现某些高级功能(如热修复、动态修改代码)非常有用。
Can-Retransform-Classes: true 这个属性允许Java Agent在运行时重新转换已加载的类。与Can-Redefine-Classes不同,Retransform要求类的原始字节码仍然可用,并且重新转换的字节码必须与原始字节码在结构上是兼容的(即具有相同的方法签名和字段)。这允许Agent在保持类结构一致性的情况下修改类的行为。
Can-Set-Native-Method-Prefix: true 这个属性允许Java Agent设置原生方法(native methods)的前缀。如果设置为true,Agent可以为加载的类中的原生方法名添加指定的前缀。这可以用于拦截或修改对原生方法的调用。然而,请注意,这个特性在某些Java版本和平台上可能不可用或受到限制。

步骤四:使用Maven Assembly插件进行打包

在命令行或IDE中使用maven-assembly-plugin插件进行打包。插件会将项目的所有依赖和编译后的类文件打包成一个JAR文件,该文件将作为Java Agent的部署包。

image.gif

步骤五:集成Java Agent到Spring Boot应用中

创建一个新的Spring Boot项目或使用现有的Spring Boot应用。在Spring Boot应用的启动脚本或配置文件中,配置Java Agent的静态加载。这需要在JVM启动参数中添加-javaagent选项,并指定Java Agent的JAR文件路径。

java -jar -javaagent:./agent.jar test.jar
image.gif
-javaagent: 告诉JVM在启动应用程序之前加载并应用这个Agent
./agent.jar Java Agent JAR文件的路径
test.jar 要运行的Java应用程序的JAR文件

2.搭建Java Agent动态加载模式的案例

步骤一:创建Maven项目并配置打包插件

首先,创建一个新的Maven项目,在项目的pom.xml文件中,添加maven-assembly-plugin插件的配置。这个插件的主要作用是将项目依赖和编译后的类文件打包成一个单独的JAR文件,该文件将作为Java Agent的部署包。

<plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-assembly-plugin</artifactId>
                <configuration>
                   <!-- 将所有依赖打入同一个jar包中 -->
                    <descriptorRefs>
                        <descriptorRef>jar-with-dependencies</descriptorRef>
                    </descriptorRefs>
                    <!--指定java agent相关配置文件-->
                    <archive>
                        <manifestFile>src/main/resources/MANIFEST.MF</manifestFile>
                    </archive>
                </configuration>
            </plugin>
image.gif

步骤二:编写Java Agent的核心代码

在项目中创建一个新的Java类,用于实现Java Agent的逻辑。在该类中编写agentmain方法。与premain方法不同,agentmain方法支持在Java应用程序运行时动态加载Agent。在这个方法中,可以编写自定义的逻辑,例如打印一行日志信息,以验证Agent已成功加载。

public class AgentMain {
    // agentmain方法
    public static void agentmain(String agentArgs, Instrumentation inst){
        System.out.println("agentmain方法");
    }
}
image.gif

步骤三:配置Java Agent的清单文件

在MANIFEST.MF文件中,指定使用哪个类的agentmain方法作为Java Agent的入口点。这告诉JVM在动态加载Agent时应该调用哪个方法。

Manifest-Version: 1.0
Agent-Class: com.rye.javaagent.AgentMain
Can-Redefine-Classes: true
Can-Retransform-Classes: true
Can-Set-Native-Method-Prefix: true
image.gif

步骤四:构建和打包Java Agent

在命令行或IDE中使用maven-assembly-plugin插件进行打包。插件会将项目的所有依赖和编译后的类文件打包成一个JAR文件,该文件将作为Java Agent的部署包。

image.gif

步骤五:编写Java应用程序以动态加载Java Agent

创建一个新的Java类或在现有的Java项目中添加一个类,用于编写动态加载Java Agent的逻辑。在该类中编写main方法,并使用Java的VirtualMachine类来连接到正在运行的Java应用程序。通过调用VirtualMachine.loadAgent方法,可以动态地将打包好的Java Agent JAR文件加载到目标应用程序中。在加载Agent之前,确保目标Java应用程序已经启动并且处于可连接状态。这涉及到设置适当的JVM参数以启用远程调试或JMX(Java Management Extensions)连接。一旦Agent成功加载,它将开始执行在agentmain方法中定义的逻辑,从而可以在不重启目标应用程序的情况下修改或监视其行为。

public class AttachMain {
    public static void main(String[] args) throws IOException, AttachNotSupportedException, AgentLoadException, AgentInitializationException {
        // 获取进程虚拟机对象
        VirtualMachine vm=VirtualMachine.attach("进程ID");
        // 执行java agent里边的agentmain方法
        vm. loadAgent("./agent.jar");
    }
}
image.gif

总结

本文探讨了Java Agent技术的核心原理和应用价值,展示了它在Java工具开发中的关键作用。通过静态加载和动态加载两种模式,Java Agent能够灵活地在Java程序的不同阶段执行自定义代码,为性能监控、故障诊断和动态代码修改提供了有力支持。希望对大家有所帮助。

相关文章
|
1月前
|
数据采集 监控 Oracle
GraalVM 24 正式发布阿里巴巴贡献重要特性 —— 支持 Java Agent 插桩
阿里巴巴是 GraalVM 全球顾问委员会的唯一中国代表,阿里云程序语言与编译器团队和可观测团队合作实现了 GraalVM 应用的无侵入可观测能力,并在 ARMS 平台上线了该功能。目前在 GraalVM 24 中发布的是支持 Java agent 的第一步,其余能力将在 GraalVM 的后续版本中陆续发布。
169 21
|
1月前
|
监控 Java Unix
6个Java 工具,轻松分析定位 JVM 问题 !
本文介绍了如何使用 JDK 自带工具查看和分析 JVM 的运行情况。通过编写一段测试代码(启动 10 个死循环线程,分配大量内存),结合常用工具如 `jps`、`jinfo`、`jstat`、`jstack`、`jvisualvm` 和 `jcmd` 等,详细展示了 JVM 参数配置、内存使用、线程状态及 GC 情况的监控方法。同时指出了一些常见问题,例如参数设置错误导致的内存异常,并通过实例说明了如何排查和解决。最后附上了官方文档链接,方便进一步学习。
|
1月前
|
前端开发 Java 物联网
智慧班牌源码,采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署
智慧班牌系统是一款基于信息化与物联网技术的校园管理工具,集成电子屏显示、人脸识别及数据交互功能,实现班级信息展示、智能考勤与家校互通。系统采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署与私有化定制。核心功能涵盖信息发布、考勤管理、教务处理及数据分析,助力校园文化建设与教学优化。其综合性和可扩展性有效打破数据孤岛,提升交互体验并降低管理成本,适用于日常教学、考试管理和应急场景,为智慧校园建设提供全面解决方案。
229 70
|
1月前
|
人工智能 Java 程序员
Java程序员在AI时代必会的技术:Spring AI
在AI时代,Java程序员需掌握Spring AI技术以提升竞争力。Spring AI是Spring框架在AI领域的延伸,支持自然语言处理、机器学习集成与自动化决策等场景。它简化开发流程,无缝集成Spring生态,并提供对多种AI服务(如OpenAI、阿里云通义千问)的支持。本文介绍Spring AI核心概念、应用场景及开发步骤,含代码示例,助你快速入门并构建智能化应用,把握AI时代的机遇。
|
2月前
|
人工智能 自然语言处理 前端开发
从理论到实践:使用JAVA实现RAG、Agent、微调等六种常见大模型定制策略
大语言模型(LLM)在过去几年中彻底改变了自然语言处理领域,展现了在理解和生成类人文本方面的卓越能力。然而,通用LLM的开箱即用性能并不总能满足特定的业务需求或领域要求。为了将LLM更好地应用于实际场景,开发出了多种LLM定制策略。本文将深入探讨RAG(Retrieval Augmented Generation)、Agent、微调(Fine-Tuning)等六种常见的大模型定制策略,并使用JAVA进行demo处理,以期为AI资深架构师提供实践指导。
335 73
|
3月前
|
JavaScript 安全 Java
智慧产科一体化管理平台源码,基于Java,Vue,ElementUI技术开发,二开快捷
智慧产科一体化管理平台覆盖从备孕到产后42天的全流程管理,构建科室协同、医患沟通及智能设备互联平台。通过移动端扫码建卡、自助报道、智能采集数据等手段优化就诊流程,提升孕妇就诊体验,并实现高危孕产妇五色管理和孕妇学校三位一体化管理,全面提升妇幼健康宣教质量。
81 12
|
2月前
|
存储 监控 数据可视化
SaaS云计算技术的智慧工地源码,基于Java+Spring Cloud框架开发
智慧工地源码基于微服务+Java+Spring Cloud +UniApp +MySql架构,利用传感器、监控摄像头、AI、大数据等技术,实现施工现场的实时监测、数据分析与智能决策。平台涵盖人员、车辆、视频监控、施工质量、设备、环境和能耗管理七大维度,提供可视化管理、智能化报警、移动智能办公及分布计算存储等功能,全面提升工地的安全性、效率和质量。
|
2月前
|
Arthas 监控 Java
拥抱 OpenTelemetry:阿里云 Java Agent 演进实践
拥抱 OpenTelemetry:阿里云 Java Agent 演进实践
106 0
|
3月前
|
缓存 Java 物联网
CRaC技术助力ACS上的Java应用启动加速
容器计算服务借助ACS的柔性算力特性并搭配CRaC技术极致地提升Java类应用的启动速度。
|
4月前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
180 7

热门文章

最新文章