一、背景
很久之前就有一个想法:如果部署在线上的代码在遇到类似空指针异常时能自动修复就好了。前段时间准备尝试落实这个想法。那就先从基础做起:先在线改代码吧!
本来我是在学习字节码的,偶尔搜了一下资料,发现Arthas这个工具,能够实现代码热修改。
我仿佛发现了新大陆,于是添加收藏。这时我发现,我的收藏夹里很早就收藏了这个网页。看来有些路终究是要走的。
废话不说,试试吧!
二、开始
1、启动springboot项目
2、运行arthas,并选择目标进程
我这里选择2
java -jar arthas-boot.jar
[INFO] arthas-boot version: 3.6.0
[INFO] Found existing java process, please choose one and input the serial number of the process, eg : 1. Then hit ENTER.
3、使用sc命令查找要修改的类
我这里准备在StringUtil里加上一个打印,输入查找命令如下:
[arthas@6787]$ sc -d *util.StringUtil
class-info com.shuimutong.learn.arthos.helloweb.util.StringUtil
code-source /Users/zhengxingao/Documents/develop/arthoslearn/helloweb/target/classes/
name com.shuimutong.learn.arthos.helloweb.util.StringUtil
isInterface false
isAnnotation false
isEnum false
isAnonymousClass false
isArray false
isLocalClass false
isMemberClass false
isPrimitive false
isSynthetic false
simple-name StringUtil
modifier public
annotation
interfaces
super-class ±java.lang.Object
class-loader ±sun.misc.Launcher A p p C l a s s L o a d e r @ 18 b 4 a a c 2 + − s u n . m i s c . L a u n c h e r AppClassLoader@18b4aac2 +-sun.misc.Launcher AppClassLoader@18b4aac2+−sun.misc.LauncherExtClassLoader@32a1bec0
classLoaderHash 18b4aac2
4、使用jad命令反编译出源码
jad --source com.shuimutong.learn.arthos.helloweb.util.StringUtil > /tmp/StringUtil.java
输出目录建议选择/tmp,不用担心没有权限
5、编辑/tmp/StringUtil.java
arthas里面没法使用vi,所以新开窗口编辑吧。
复制代码
/*
* Decompiled with CFR.
*/
package com.shuimutong.learn.arthos.helloweb.util;
public class StringUtil {
public static String packHello(String name) {
System.out.println("name:"+name);
/* 5*/ return String.format(“Hello %s!”, name);
}
}
复制代码
我加了个打印。
6、编译文件
mc -c 18b4aac2 /tmp/StringUtil.java -d /tmp
Memory compiler output:
/tmp/com/shuimutong/learn/arthos/helloweb/util/StringUtil.class
Affect(row-cnt:1) cost in 194 ms.
-c后面跟的一串是字符是前面查找类时打印出的classLoadHash
7、加载编译的class
[arthas@6787]$ redefine /tmp/com/shuimutong/learn/arthos/helloweb/util/StringUtil.class
redefine success, size: 1, classes:
com.shuimutong.learn.arthos.helloweb.util.StringUtil
8、检验
请求:http://localhost:8081/hello1?name=arthas
控制台打印:name:arthas
符合预期。
热更新注意事项
redefine 特别说明
redefine 命令和 jad/watch/trace/monitor/tt 等命令会冲突。
执行完 redefine 之后,如果再执行上面提到的命令,则会把 redefine 的字节码重置。
原因是 jdk 本身 redefine 和 Retransform 是不同的机制,同时使用两种机制来更新字节码,只有最后修改的会生效。
条件限制
使用热更新功能有一些条件限制,我们只能用它来修改方法内部的一些业务代码,如果我们出现了以下任意一种情况,那么热更新就会执行失败:
增加类属性(类字段);
增加或删除方法;
替换正在运行的方法。