Java动态替换class:agentmain + attach + transformer

本文介绍了如何使用阿里开发的Arthas工具动态介入Java程序。通过创建一个基础项目MyClass,然后编写一个agent项目包含AgentMain和Transformer,动态添加和移除ClassFileTransformer来实现对MyClass的修改。最后,创建一个Attach项目来连接并注入agent到MyClass进程中,实现了运行时替换类的行为。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前段时间,用了阿里开发的arthas,对于它是如何动态介入java程序的很感兴趣,最近有时间,就自己研究了一下,写了个简单的项目试了试。

项目结构

在这里插入图片描述

如图所示,总共分为三个项目,分别是:

  • 初时的启动项目,就是我们想要动态修改的目标项目MyClass;
  • 将被attach到目标的agent项目;
  • 负责将agent项目attach到目标项目的attach项目;

MyClass项目

只有一个类文件:Main.java

package org.example;

public class Main {
    public static void greet() {
        System.out.println("Hello!");
    }

    public static void main(String[] args) throws InterruptedException {
        while (true) {
            greet();
            Thread.sleep(1000 * 3);
        }
    }
}

因为要打包,需要在pom文件中添加plugin:

<build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-jar-plugin</artifactId>
                <version>2.3.2</version>
                <configuration>
                    <archive>
                        <manifestFile>src/main/resources/MANIFEST.MF</manifestFile>
                    </archive>
                </configuration>
            </plugin>
        </plugins>
    </build>

MANIFEST.MF文件:

Main-Class: org.example.Main

然后使用Maven打包运行即可。
该项目每隔三秒输出一个"Hello!"。

Agent项目

AgentMain文件:

package org.agent;

import java.lang.instrument.Instrumentation;
import java.lang.instrument.UnmodifiableClassException;

public class AgentMain {
    public static void agentmain(String agentArgs, Instrumentation inst) throws UnmodifiableClassException,
            ClassNotFoundException {

        Transformer transformer = new Transformer(agentArgs);
        try {
            inst.addTransformer(transformer, true);
            inst.retransformClasses(Class.forName("org.example.Main"));
        } finally {
            new Thread(() -> {
                try {
                    Thread.sleep(10 * 1000);
                    inst.removeTransformer(transformer);
                    inst.retransformClasses(Class.forName("org.example.Main"));
                } catch (Exception e) {}
            }).start();
        }
    }
}

Transformer文件:

package org.agent;

import java.io.FileInputStream;
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.security.ProtectionDomain;

public class Transformer implements ClassFileTransformer {

    private String classPath;

    public Transformer(String classPath) {
        this.classPath = classPath;
    }

    @Override
    public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {
        className = className.replace("/", ".");
        if (className.equals("org.example.Main")) {
            System.out.println("retransform myclass:" + classPath);
            FileInputStream inputStream = null;
            try {
                inputStream = new FileInputStream(classPath);
                byte[] bytes = new byte[inputStream.available()];
                inputStream.read(bytes);
                return bytes;
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }
        return classfileBuffer;
    }
}

因为要打包,需要在pom文件中添加plugin:

<build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-jar-plugin</artifactId>
                <version>2.3.2</version>
                <configuration>
                    <archive>
                        <manifestFile>src/main/resources/MANIFEST.MF</manifestFile>
                    </archive>
                </configuration>
            </plugin>
        </plugins>
    </build>

MANIFEST.MF文件:

Agent-Class: org.agent.AgentMain
Can-Retransform-Classes: true

注意这里配置的是Agent-Class,用的是AgentMain,可以动态Attach;之前试过PrevMain,那个不是动态的,启动时需要指定。
该Agent主要是添加一个transformer,然后retransform目标class,过个十秒再删除掉之前的transformer,将替换的类恢复。

transformer根据Attach提供的参数,读取class文件,替换旧的目标class。
打包获得一个agent-1.0-SNAPSHOT.jar文件。

用于替换的Main文件:

package org.example;

public class Main {
    public static void greet() {
        System.out.println("Hi!");
    }

    public static void main(String[] args) throws InterruptedException {
        while (true) {
            greet();
            Thread.sleep(1000 * 3);
        }
    }
}

注意,包名和类名都要和源文件一致,不然会报错。这个文件其实不需要放在Agent项目里面,我只是为了方便,随便放在这里的,自己可以使用javac编译一下,得到Main.class。

Attach项目

Attach项目只有一个Main类,也不需要打包,直接运行就行。

package org.example;

import com.sun.tools.attach.*;

import java.io.IOException;
import java.util.List;

public class Main {

    //agent路径和新的class的路径,自己替换即可
    public static final String AGENT_PATH = "***/agent/target/agent-1.0-SNAPSHOT.jar";
    public static final String NEW_CLASS_PATH = "***/agent/src/main/java/org/example/Main.class";

    public static void main(String[] args) throws IOException, AttachNotSupportedException, AgentLoadException,
            AgentInitializationException, InterruptedException {

        // 获取正在运行的JVM的列表
        List<VirtualMachineDescriptor> list = VirtualMachine.list();
        
        for (VirtualMachineDescriptor descriptor : list) {
            // 根据进程名字获取进程ID, 并使用 loadAgent 注入进程
            if (descriptor.displayName().contains("MyClass")) {
                VirtualMachine virtualMachine = VirtualMachine.attach(descriptor.id());
                virtualMachine.loadAgent(AGENT_PATH, NEW_CLASS_PATH);
                virtualMachine.detach();
            }
        }
    }
}

运行结果

先启动MyClass的jar包,然后运行Attach项目的Main类,即可得到如下的结果:
在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值