JAVA代码的热修复实现

简介: JAVA代码的热修复实现

用于处理线上的一些逻辑bug的利器,这样,就不会为了一些几行代码的错误导致的bug,需要重新发版本重启服务器,而严重影响在线活跃,收入了。

1.实现一个代理类:

package com.lingyu.common.hotcode;
import java.lang.instrument.ClassDefinition;
import java.lang.instrument.Instrumentation;
/**
 * java 代码热更新
 */
public class HotCodeAgent {
private static final Logger logger = LogManager.getLogger(HotCodeAgent.class);
  public static Instrumentation INST;
  /** JVM 首先尝试在代理类上调用以下方法 */
  public static void premain(String agentArgs, Instrumentation inst) {
    INST = inst;
  }
  /**
   * 重新定义可能会改变方法体,常量池和属性。重定义不能添加,删除或重命名字段或方法,更改方法的签名或更改继承。这些限制可能在将来的版本中解除。
   * 类文件字节不会被检查,验证和安装,直到应用转换为止,如果结果字节错误,则此方法将抛出异常。 如果此方法抛出异常,则不会重新定义任何类。
   */
  public static void reload(Class<?> clazz, byte[] data) {
     try {
      ClassDefinition classDefinition = new ClassDefinition(clazz, data);
      INST.redefineClasses(classDefinition);
    } catch (Throwable e) {
      logger.error(e.getMessage(), e);
    }   
  }
}

2.在game-common的build.xml里增加节点:

<manifest file="MANIFEST.MF">
  <attribute name="Built-By" value="${user.name}" />
  <attribute name="Built-Date" value="${buildTime}" />
  <attribute name="Implementation-Version" value="${branch}-${svn}-${version}" />
  <attribute name="Premain-Class" value="com.lingyu.common.hotcode.HotCodeAgent" />
  <attribute name="Can-Redefine-Classes" value="true" />
</manifest>

3.在启动参数中加入参数 java -javaagent:lib/game-common.jar -server -Xms1024m -Xmx1024m -jar game.jar

游戏服务器启动的时候 会根据game-common.jar里的manifest.mf文件查找premain-class调用premain方法 项目保留Instrumentation的引用。

4. 通过后台上传class文件 更新到线上 建议 先在内网环境测试,一般只热更新逻辑代码。

class文件-->通过后台上传-->把class字节数据分发给线上,解析,重新加载类定义。(该画个图)

public void handleHotCode(String message) {
    // TODO Auto-generated method stub
    HotCodeInfo info = JSON.parseObject(message,HotCodeInfo.class);
    Class<?> clazz = null;
    try {
      //完整的包名类路径直接获取到已经加载的类
      if(StringUtils.isNotEmpty(info.getReplaceName())){
        clazz = Class.forName(info.getReplaceName());
      }
      else{
        //直接根据spring的接口获取对应的类
        clazz = GameServerContext.getBean(info.getName()).getClass();
      }
      HotCodeAgent.reload(clazz,info.getData());
    } catch (Exception e) {
      logger.error(e.getMessage(),e);
    }
  }

最近遇到个问题,有没有遇到过热加载时有内部类或者匿名内部类的类 ,加载的时候没法成功。抛出的异常为:

2018-12-19 10:36:16.000 ERROR [Message SubScribe Monitor][HotCodeAgent] - class redefinition failed: attempted to add a method
java.lang.UnsupportedOperationException: class redefinition failed: attempted to add a method
        at sun.instrument.InstrumentationImpl.redefineClasses0(Native Method)
        at sun.instrument.InstrumentationImpl.redefineClasses(InstrumentationImpl.java:170)
        at com.lingyu.common.hotcode.HotCodeAgent.reload(HotCodeAgent.java:53)

或者:

java.lang.UnsupportedOperationException: class redefinition failed: attempted to change the schema (add/remove fields)

同样的一份class文件,为什么会提示新增或者删除了方法呢?

然后我们开始对比 javap -p ooxx.class ,发现在底部多了这些方法,这些都是内部类访问外部类的变量或者方法时调用导致的。

static java.util.Map access$0(com.lingyu.game.service.role.RoleManager);
  static org.apache.logging.log4j.Logger access$1();

只要规避掉这样的调用,就可以让含有内部类的类热加载。

同事还是很努力的!点个赞!

目录
相关文章
|
1月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
67 1
|
1月前
|
Java
在Java中实现接口的具体代码示例
可以根据具体的需求,创建更多的类来实现这个接口,以满足不同形状的计算需求。希望这个示例对你理解在 Java 中如何实现接口有所帮助。
87 38
|
4天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
20 3
|
1月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
52 24
|
12天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
49 2
|
26天前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
66 5
|
26天前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
53 5
|
28天前
|
Java API 开发者
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
|
1月前
|
Java API Maven
商汤人像如何对接?Java代码如何写?
商汤人像如何对接?Java代码如何写?
41 5
|
1月前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
47 5
Java反射机制:解锁代码的无限可能