Java Agent是什么?
Java Agent 又叫做 Java 探针,是在 JDK1.5 引入的一种可以动态修改 Java 字节码的技术。Java 类编译之后形成字节码被 JVM 执行,在 JVM 在执行这些字节码之前获取这些字节码信息,并且通过字节码转换器对这些字节码进行修改,来完成一些额外的功能。java agent本质上可以理解为一个jar包插件,这个jar包通过JVMTI(JVM Tool Interface,是Java虚拟机对外提供的Native编程接口,外部进程可以获取到运行时JVM的很多信息,比如线程、GC等)完成加载,利用JVM提供的Instrumentation API可以在 JVM 执行字节码之前或在JVM 加载字节码之后对字节码进行修改。-javaagent是java命令的一个参数,应用启动时可以用这个参数指定一个jar包,去实现我们想做事情。
Java Agent能做什么?
可以在加载class文件之前做拦截,对字节码做修改
可以在运行期将已经加载的类的字节码做变更
获取所有已经被加载过的类
获取所有已经被初始化过了的类
获取某个对象的大小
将某个jar加入到bootstrapclasspath里作为高优先级被bootstrapClassloader加载
将某个jar加入到classpath里供AppClassloard去加载
设置某些native方法的前缀,主要在查找native方法的时候做规则匹配
链路追踪系统:skywalking
诊断工具:Arthas
如果要使用agent,我们需要定义两个方法
premain: 将在JVM启动时使用-javaagent参数静态地加载agent
agentmain: 将使用Java Attach API动态加载agent到JVM中
package org.example.agent;
import java.lang.instrument.Instrumentation;
public class MyAgent {
/**
* jvm参数形式调用premain方法,在java程序的main方法执行之前执行
*/
public static void premain(String agentArgs, Instrumentation inst) {
System.out.println("[MyAgent][premain]: " + agentArgs);
System.out.println("Premain-Class: " + MyAgent.class.getName());
System.out.println("Can-Redefine-Classes: " + inst.isRedefineClassesSupported());
System.out.println("Can-Retransform-Classes: " + inst.isRetransformClassesSupported());
System.out.println("Can-Set-Native-Method-Prefix: " + inst.isNativeMethodPrefixSupported());
System.out.println("========= ========= =========");
}
/**
* attach方式调用agentmain方法,在java程序启动后执行
*/
public static void agentmain(String agentArgs, Instrumentation inst) {
System.out.println("[MyAgent][agentmain]: " + agentArgs);
System.out.println("========= ========= =========");
}
}

Manifest-Version: 1.0
Premain-Class: org.example.agent.MyAgent
Agent-Class: org.example.agent.MyAgent
Can-Redefine-Classes: true
Can-Retransform-Classes: true
Can-Set-Native-Method-Prefix: true

Premain-Class:包含premain方法的类,需要配置为类的全路径
Agent-Class:包含agent方法的类,需要配置为类的全路径
Can-Redefine-Classes:为true时表示能够重新定义class
Can-Retransform-Classes:为true时表示能够重新转换class,实现字节码替换
Can-Set-Native-Method-Prefix:为true时表示能够设置native方法的前缀
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0modelVersion>
<groupId>org.examplegroupId>
<artifactId>my-agentartifactId>
<version>1.0-SNAPSHOTversion>
<name>my-agentname>
<url>http://www.example.comurl>
<properties>
<project.build.sourceEncoding>UTF-8project.build.sourceEncoding>
<maven.compiler.source>8maven.compiler.source>
<maven.compiler.target>8maven.compiler.target>
properties>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.pluginsgroupId>
<artifactId>maven-assembly-pluginartifactId>
<configuration>
<descriptorRefs>
<descriptorRef>jar-with-dependenciesdescriptorRef>
descriptorRefs>
<archive>
<manifestFile>src/main/resources/META-INF/MANIFEST.MFmanifestFile>
archive>
configuration>
plugin>
plugins>
build>
project>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0modelVersion>
<groupId>org.examplegroupId>
<artifactId>my-agentartifactId>
<version>1.0-SNAPSHOTversion>
<name>my-agentname>
<url>http://www.example.comurl>
<properties>
<project.build.sourceEncoding>UTF-8project.build.sourceEncoding>
<maven.compiler.source>8maven.compiler.source>
<maven.compiler.target>8maven.compiler.target>
properties>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.pluginsgroupId>
<artifactId>maven-assembly-pluginartifactId>
<configuration>
<descriptorRefs>
<descriptorRef>jar-with-dependenciesdescriptorRef>
descriptorRefs>
<archive>
<manifestEntries>
<Premain-Class>org.example.agent.MyAgentPremain-Class>
<Agent-Class>org.example.agent.MyAgentAgent-Class>
<Can-Redefine-Classes>trueCan-Redefine-Classes>
<Can-Retransform-Classes>trueCan-Retransform-Classes>
<Can-Set-Native-Method-Prefix>trueCan-Set-Native-Method-Prefix>
manifestEntries>
archive>
configuration>
plugin>
plugins>
build>
project>


<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0modelVersion>
<parent>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-parentartifactId>
<version>2.6.6version>
<relativePath/>
parent>
<groupId>com.examplegroupId>
<artifactId>maven-demoartifactId>
<version>0.0.1-SNAPSHOTversion>
<name>maven-demoname>
<description>maven-demodescription>
<properties>
<java.version>8java.version>
properties>
<dependencies>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
<executions>
<execution>
<goals>
<goal>repackagegoal>
goals>
execution>
executions>
<configuration>
<includeSystemScope>trueincludeSystemScope>
<mainClass>com.example.mavendemo.MavenDemoApplicationmainClass>
configuration>
plugin>
plugins>
build>
project>

package com.example.mavendemo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MavenDemoApplication {
public static void main(String[] args) {
System.out.println("hello world!");
SpringApplication.run(MavenDemoApplication.class, args);
System.out.println("hahaha");
}
}

-javaagent:xxx.jar参数

${JAVA_8_HOME}/lib/tools.jar就是jdk8下面com.sun.tools存放的位置,其他版本的jdk不一定是在这个文件夹下,${JAVA_8_HOME}是你自己设置jdk环境变量时的名字,根据自己的设置进行修改
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0modelVersion>
<groupId>org.examplegroupId>
<artifactId>attach-demoartifactId>
<version>1.0-SNAPSHOTversion>
<name>attach-demoname>
<url>http://www.example.comurl>
<properties>
<project.build.sourceEncoding>UTF-8project.build.sourceEncoding>
<maven.compiler.source>8maven.compiler.source>
<maven.compiler.target>8maven.compiler.target>
properties>
<dependencies>
<dependency>
<groupId>com.sungroupId>
<artifactId>toolsartifactId>
<version>8version>
<scope>systemscope>
<systemPath>${JAVA_8_HOME}/lib/tools.jarsystemPath>
dependency>
dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.pluginsgroupId>
<artifactId>maven-assembly-pluginartifactId>
<version>2.5.5version>
<configuration>
<archive>
<manifest>
<mainClass>org.example.attach.AttachermainClass>
manifest>
archive>
<descriptorRefs>
<descriptorRef>jar-with-dependenciesdescriptorRef>
descriptorRefs>
configuration>
plugin>
plugins>
build>
project>


package org.example.attach;
import java.io.IOException;
import com.sun.tools.attach.AgentInitializationException;
import com.sun.tools.attach.AgentLoadException;
import com.sun.tools.attach.AttachNotSupportedException;
import com.sun.tools.attach.VirtualMachine;
public class Attacher {
public static void main(String[] args)
throws IOException, AttachNotSupportedException, AgentLoadException, AgentInitializationException {
// 指定进程号
VirtualMachine vm = VirtualMachine.attach(args[0]);
// 指定agent包位置和要传入的参数
vm.loadAgent(
"/Users/fisher/Documents/code/gitee/myAgent/target/my-agent-1.0-SNAPSHOT-jar-with-dependencies.jar",
"Hello JVM Attach");
vm.detach();
}
}



java -javaagent:/Users/xxx/Documents/code/gitee/myAgent/target/my-agent-1.0-SNAPSHOT-jar-with-dependencies.jar=Fisher -jar maven-demo-0.0.1-SNAPSHOT.jar启动,同样可以看到premain中的方法被执行了
jps
java -jar attach-demo-1.0-SNAPSHOT-jar-with-dependencies.jar 2555,2555就是待修改的运行中的程序的进程号

package org.example.attach;
import java.io.IOException;
import java.util.List;
import com.sun.tools.attach.AgentInitializationException;
import com.sun.tools.attach.AgentLoadException;
import com.sun.tools.attach.AttachNotSupportedException;
import com.sun.tools.attach.VirtualMachine;
import com.sun.tools.attach.VirtualMachineDescriptor;
public class Attacher {
public static void main(String[] args)
throws IOException, AttachNotSupportedException, AgentLoadException, AgentInitializationException {
// 查询所有运行的java程序
List<VirtualMachineDescriptor> list = VirtualMachine.list();
for (VirtualMachineDescriptor virtualMachineDescriptor : list) {
String displayName = virtualMachineDescriptor.displayName();
// 这里指定运行的程序的名称,使用jps就能看出名称
if (displayName.equals("maven-demo-0.0.1-SNAPSHOT.jar")) {
System.out.println(virtualMachineDescriptor.id());
VirtualMachine attach = VirtualMachine.attach(virtualMachineDescriptor);
attach.loadAgent("/Users/xxx/Documents/code/gitee/myAgent/target/my-agent-1.0-SNAPSHOT-jar-with-dependencies.jar",
"Hello VirtualMachineDescriptor");
attach.detach();
}
}
}
}

java -jar attach-demo-1.0-SNAPSHOT-jar-with-dependencies.jar
