使用Arthas实现JAVA热更新

Arthas是阿里巴巴开源的一款强大Java诊断工具,帮助开发者解决线上问题,无需重启应用即可热更新代码,实现快速迭代。本文详细介绍Arthas的安装、启动及热更新流程,分享实际操作中可能遇到的问题及解决方案。

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

Arthas是阿里巴巴开源出来的一个针对java的工具,主要是针对java的问题进行诊断。

官网地址(下载安装请看这里):https://alibaba.github.io/arthas/index.html

这个工具可以协助完成下面这些事情(转自官网):

  • 这个类是从哪个jar包加载而来的?
  • 为什么会报各种类相关的Exception?
  • 线上遇到问题无法debug好蛋疼,难道只能反复通过增加System.out或通过加日志再重新发布吗?
  • 线上的代码为什么没有执行到这里?是由于代码没有commit?还是搞错了分支?
  • 线上遇到某个用户的数据处理有问题,但线上同样无法 debug,线下无法重现。
  • 是否有一个全局视角来查看系统的运行状况?
  • 有什么办法可以监控到JVM的实时运行状态?

Arthas采用命令行交互模式,同时提供丰富的Tab自动补全功能,进一步方便进行问题的定位和诊断。

arthas实现热更新

1、启动Arthas

java -jar arthas-boot.jar

或者

java -jar arthas-boot.jar XXX  #进程的PID

2、然后选择我们需要热更新的JVM进程

[INFO] arthas-boot version: 3.1.0
[INFO] Process 58827 already using port 3658
[INFO] Process 58827 already using port 8563
[INFO] Found existing java process, please choose one and hit RETURN.
* [1]: 58827 org.apache.catalina.startup.Bootstrap
  [2]: 59288 org.apache.catalina.startup.Bootstrap
  [3]: 59482 org.apache.catalina.startup.Bootstrap
  [4]: 1857 tms-gateway-proxy-0.0.1-SNAPSHOT.jar
  [5]: 59834 org.tanukisoftware.wrapper.WrapperSimpleApp
1

在这台服务器上一共有5个java进程,可以通过ps命令确认我们使用的是哪个。这里我们选择了第一个。
一段启动信息后,就进入了交互模式。


3、通过sc查找需要修改的class的ClassLoader

$ sc -d *OAuthClient | grep classLoaderHash
classLoaderHash   452c5c14
classLoaderHash   452c5c14 

4、再使用redefine命令重新加载新编译好的OAuthClient.class

$ redefine -c 452c5c14  /tmp/OAuthClient.class
redefine success, size: 1

注意:不允许新增加field/method,正在跑的函数,没有退出不能生效

其他热更新操作方法:

jad --source-only com.example.demo.arthas.user.UserController > /tmp/UserController.java 
mc /tmp/UserController.java -d /tmp 
redefine /tmp/com/example/demo/arthas/user/UserController.class
  • jad命令反编译,然后可以用其它编译器,比如vim来修改源码
  • mc命令来内存编译修改过的代码
  • 用redefine命令加载新的字节码

JVM热更新的局限

基于Attach机制实现的热更新,更新类需要与原来的类在包名,类名,修饰符上完全一致,否则在classRedefine过程中会产生classname don't match 的异常。

例如显示这样的报错:redefineClasses exception class redefinition failed: attempted to delete a method.

具体来说,JVM热更新局限总结:

  1. 函数参数格式不能修改,只能修改函数内部的逻辑
  2. 不能增加类的函数或变量
  3. 函数必须能够退出,如果有函数在死循环中,无法执行更新类(笔者实验发现,死循环跳出之后,再执行类的时候,才会是更新类)

遇到的问题及解决:(大部分问题可以在Arthas上面找到解决方案)

1、启动报错,no attach in java.library.path

原因:java启动环境不对,我的机器只装了运行环境jre,少了很多东西

解决方法:重新下载了JDK,使用JDK的java运行

/root/jdk1.7.0_80/bin/java -jar arthas-boot.jar

2、attach报错,well-known file is not secure

原因:用root用户启动,去attach其他进程,不被允许

解决方法:切换的对应用户启动

参考:

1、https://www.jianshu.com/p/8c4f8762389e 更多其他热更新方法

2、https://www.cnblogs.com/jpfss/p/10943731.html  arthas安装,启动,热更新

3、https://www.jianshu.com/p/8fe746b871a9 arthas其他更多有用的用法

### 如何使用 Arthas 进行热更新 Arthas 是阿里巴巴开源的一款强大的 Java 诊断工具,能够帮助开发者在线调试生产环境中的问题。通过 Arthas 的 `redefine` 命令可以实现类的热更新功能。 以下是关于如何使用 Arthas 实现热更新以及与 `null` 使用的相关说明: #### 1. 启动 Arthas 并连接目标 JVM 启动 Arthas 工具并附着到指定的目标进程上: ```bash java -jar arthas-boot.jar ``` 输入目标 JVM 的 PID 完成连接。 #### 2. 修改源码文件 在本地修改需要调整的 Java 类文件,并重新编译生成新的 `.class` 文件。 #### 3. 使用 redefine 命令加载新字节码 假设已经有一个名为 `com.example.MyClass` 的类需要更新,则可以通过以下命令完成热更新操作: ```bash redefine /path/to/MyClass.class ``` 此过程会将新的字节码动态加载至运行时环境中而无需重启应用服务[^4]。 需要注意的是,在实际开发过程中遇到涉及 `null` 处理的情况时,应特别小心防止 NullPointerException (简称 NPE),正如前面提到的标准指针编程实践所强调的一样[^2]。例如当判断某个对象是否为空之前应当先验证其状态再做进一步处理逻辑;另外对于数据库字段如果允许存在缺失值则可能将其设为 SQL 中的 NULL 表示未知或者不存在的数据项[^1]。 此外,在某些特定场景下为了保持变量生命周期跨越多次调用之间持续有效可采用 static 关键字来延长局部变量生命期[^3]。 #### 示例代码展示 下面给出一段简单的演示程序用于理解上述概念的应用方式: ```java public class MyClass { public void sayHello() { System.out.println("Original Hello"); } } // 编辑后版本如下所示: public class MyClass { private String name; public void setName(String n){ this.name=n; } public void greet(){ if(this.name != null && !this.name.isEmpty()){ System.out.println("Hello "+name); }else{ System.err.println("Name cannot be empty or null!"); } } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值