注意:如下几点情况无法进行arthas热更新:
增加、删除,修改了 field(字段) ;
- attempted to add Or remove field;
增加、删除了 method (方法);
- redefine error! java.lang.UnsupportedOperationException: class redefinition failed: attempted to add a method
替换正在运行的方法;
编译时会报错:
目标类中存在Lombok,可在IDEA中直接refactor-》DeLombok-》all-lombok-annotations来将类中的lombok去除;
不要给抽象父类的方法加上@override不然还是得重新打包并重启,不能做热更新;
实验目标:由于业务逻辑变更,对目标类进行修改,并热更新上线;
这里就以正常的问题排查流程来走了;
- 发现部分接口出现问题,启动arthas;
# 下载arthas的包
wget https://arthas.aliyun.com/arthas-boot.jar
# 启动
java -jar arthas-demo.jar
- 查询对应请求的controller层并查看其异常
# 查看请求参数和堆栈 支持通配符 *
# watch [类路径] [方法名称] [返回值表达式] [-x 2 将结果参数展开]
watch com.xxxxx.server.component* * '{params,throwExp}' -x 2
返回值表达式中它实际上是一个ognl表达式,它支持一些内置对象:loader、clazz、method、target、params、returnObj、throwExp、isBefore、isThrow、isReturn
# 这里随便写一个来进行查询
# 回显如下
[arthas@14016]$ watch com.xxxxx.server.component.xxxxxComponent * '{params,throwExp}' -x 2
Press Q or Ctrl+C to abort.
Affect(class count: 2 , method count: 61) cost in 191 ms, listenerId: 4
method=com.xxxxx.server.component.xxxxxComponent.activateCaChe location=AtExit
ts=2022-09-20 16:53:18; [cost=0.1266ms] result=@ArrayList[
@Object[][
@String[参数详情],
@String[参数详情],
],
null,
]
回显参数会以数组的形式展示出来:result=@ArrayList[]
- 假定现在我们知道出错是什么原因了,将目标类进行反编译,并将java文件放到对应文件夹目录下(也可以跳过此步直接使用项目中现成的java类进行后续操作)
jad --source-only com.xxxxx.server.component.xxxxxComponent > /tmp/xxxxxComponent.java
- 根据修改规则以及注意事项修改目标类,并进行重新编译;
# mc [--classLoaderClass 指定的类加载器(可通过classLoaderHash代替)] [要修改的参数路径] [-d 设置class文件的目标目录(后续redefine要用到)]
mc --classLoaderClass org.springframework.boot.loader.LaunchedURLClassLoader /tmp/xxxxxComponent.java -d /tmp
- 热更新;
# 重新定义类 redefine [编译后的类路径]
redefine /tmp/com/xxxxx/server/component/xxxxxComponent.class
- 校验修改结果;