动态热更新线上代码(Arthas)

王大爷 2022年07月13日 432次浏览

一、背景

有时我们在项目中正在运行的线上代码出了问题,由于是线上项目无法重启或更新代码,此时只能选择热更新的方式进行,但是热更新有风险,操作需谨慎,一般用来解决部署难度高的环境,动态debug问题
这里:默认大家已经会使用arthas或者arthas已经安装完成,如果没有安装请点击:Arthas-java在线调试工具的使用

二、步骤

测试代码:

package com.tiandy.testdemo;
@SpringBootApplication
public class TestDemoApplication {

    public static void main(String[] args) throws InterruptedException {
        SpringApplication.run(TestDemoApplication.class, args);
        all();

    }
    public static void all() throws InterruptedException {
        int space = 2;
        do {
            add(space);
        } while (true);
    }

    static int i = 0;

    public static void add(int space) throws InterruptedException {
        TimeUnit.SECONDS.sleep(1);
        System.out.println("★★★★★★---》" + (i += space));
    }

}

过程

## 找到项目进程号
ps -ef | grep TestDemo
## 启动arthas 
#/opt/apache-tomcat/jre/bin/java 是我java的安装位置
#/usr/arthas/arthas-boot.jar 是arthas的安装位置
/opt/apache-tomcat/jre/bin/java -jar /usr/arthas/arthas-boot.jar 21932
## 通过jad反编译代码,把它保存到另一个.java 文件中
## 注意--source-only 要进跟在jad后面
jad  --source-only  com.tiandy.testdemo.TestDemoApplication > /var/TestDemoApplication.java
## 再开一窗口编辑/var/TestDemoApplication.java,修改代码
vim /var/TestDemoApplication.java
## 查找这个类的类加载器,编译时不指定类加载器可能会失败
sc -d *TestDemoApplication | grep classLoaderHash
## 编译为class文件
mc -c de5a126b /var/TestDemoApplication.java -d /var/TestDemoApplication.class
## 加载外部的class文件,实现热更新
redefine -c de5a126b /var/com/tiandy/testdemo/TestDemoApplication.class

2.1 第一步:找到进程号

找到进程号

2.2 第二步:启动arthas

启动arthas

2.3 第三步:反编译代码,并修改

原始的

image.png

public static void all() throws InterruptedException {
        int space = 3;
        do {
            add(space);
        } while (true);
    }

    static int i = 0;

    public static void add(int space) throws InterruptedException {
        System.out.println("★★★★★★---》"+"111111111");
        TimeUnit.SECONDS.sleep(1);
        System.out.println("★★★★★★---》" + (i += space));
    }

修改3个地方

image.png

2.4 第四步: 查找这个类的类加载器的hashcode

image.png

2.5 第五步:编译java代码

image.png

2.6第六步:热更新

image.png

2.7 第七步:最终效果

image.png

三、注意事项

3.1 不允许新增加field/method

3.2 正在跑的函数,没有退出不能生效,比如下面新增加的System.out.println,只有run()函数里的会生效

3.3 如果线上的服务器不允许上传 文件,该如何解决?

第一步:在【本地】服务器先转换.class文件为base64,再保存为result.txt

base64 < Test.class > result.txt
  • 1

第二步:到【线上】服务器上,新建并编辑result.txt,复制本地的内容,粘贴再保存
第三步:把【线上】服务器上的 result.txt还原为.class

base64 -d < result.txt > Test.class