当前位置: 首页 > backend >正文

热部署与双亲委派

热部署初探与双亲委派机制

一、热部署初探

​ 热部署就是在不重启服务的情况下,无需重新启动整个应用,就能对代码、配置等进行更新并使新的更改在服务中生效。以下代码可以打破双亲委派机制,利用类加载器的隔离实现热部署。可分为以下三步进行:

  1. 自定义类加载器

    public class HotClassLoader extends ClassLoader {// 1. 指定父类加载器(默认系统类加载器)public HotClassLoader(ClassLoader parent) {super(parent);}// 2. 核心方法:从字节码加载类public Class<?> loadByte(byte[] classByte) {return defineClass(null, classByte, 0, classByte.length);}
    }
    
  2. 加载类定义

    public class HotDemo {public void printVersion() {System.out.println("【原始版本】9.0");}public static void main(String[] args) throws Exception {}
    }
    
  3. 测试类

    public class HotLoadTest {public static void main(String[] args) throws Exception {// 注意: 这是一个循环。while (true) {// 1. 读取更新后的class文件byte[] bytes = Files.readAllBytes(Paths.get("E:\\IDEAProjects\\JavaBasicKnowladge\\out\\production\\JavaBasicKnowladge\\orverLoad\\HotDemo.class"));// 2. 用自定义加载器加载HotClassLoader loader = new HotClassLoader(HotLoadTest.class.getClassLoader());Class<?> clazz = loader.loadByte(bytes);// 3. 反射调用方法Object obj = clazz.getDeclaredConstructor().newInstance();clazz.getMethod("printVersion").invoke(obj);Thread.sleep(3000); // 3秒后重新加载}}
    }
    
热部署机制:
  1. 打破类加载缓存:通常来说,类加载器会对已加载的类进行缓存,减少重复创建。上面程序每次新建HotClassLoader实例,利用不同类加载器的独立命名空间,即使类名相同。JVM也会将其视为不同的类,从而绕过缓存机制。(类加载器的命名空间(Namespace) 是JVM用于隔离不同类加载器加载的类的核心机制。JVM判断类的唯一性,通过类的全限定名与加载该类的加载器实例判断。
  2. 动态加载字节码:loadByte()方法直接调用defineClass(),这一步绕过了传统的类加载流程,将外部传入的最新字节码动态转换为Class对象。这使得修改后的类无需重启即可被加载。
  3. 隔离父类加载器:显式指定父加载器,并确保目标类未被父加载器加载过,避免双亲委派机制导致无法重新加载。
二、打破双亲委派机制
  1. 打破双亲委派机制类加载器

    public class BreakDelegateLoader extends ClassLoader {// 必须重写loadClass而非findClass@Overridepublic Class<?> loadClass(String name) throws ClassNotFoundException {// 1. 优先检查是否已加载Class<?> c = findLoadedClass(name);if (c != null) {return c;}// 2. 针对特定包名打破委派if (name.startsWith("com.example.breaking")) {return findClass(name);}// 3. 其他类仍走双亲委派return super.loadClass(name);}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {try {byte[] data = loadClassData(name);return defineClass(name, data, 0, data.length);} catch (IOException e) {throw new ClassNotFoundException(name);}}private byte[] loadClassData(String className) throws IOException {String path = className.replace('.', '/') + ".class";try (InputStream ins = getClass().getClassLoader().getResourceAsStream(path)) {// Java 8 兼容写法ByteArrayOutputStream buffer = new ByteArrayOutputStream();byte[] data = new byte[1024];int bytesRead;while ((bytesRead = ins.read(data, 0, data.length)) != -1) {buffer.write(data, 0, bytesRead);}return buffer.toByteArray();}}
    }
    
  2. 加载类

    public class TestClass {static {System.out.println("【类加载】初始化完成,加载器:" +TestClass.class.getClassLoader());}// 添加构造方法调用public TestClass() {System.out.println("实例已创建");}
    }
    
  3. 测试类

    public class BB {public static void main(String[] args) throws Exception {// 使用自定义加载器优先加载BreakDelegateLoader loader = new BreakDelegateLoader();// 必须通过自定义加载器触发首次加载Class<?> c1 = loader.loadClass("com.example.breaking.TestClass");c1.newInstance(); // 触发初始化// 再用系统加载器加载Class<?> c2 = Class.forName("com.example.breaking.TestClass");c2.newInstance();System.out.println("是否为同一个类: " + (c1 == c2)); // 应为false}
    }
    

示意图:
在这里插入图片描述

机制优势典型场景
双亲委派安全性、类唯一性、避免冲突常规Java应用、核心类库加载
打破双亲委派灵活性、隔离性、动态性Web容器、热部署、SPI、插件化架构

核心权衡:在安全稳定灵活扩展之间取舍。双亲委派是默认的“安全模式”,而打破它是为了满足特定场景下的高级需求。理解两者的优劣,才能合理设计类加载策略。

http://www.xdnf.cn/news/5783.html

相关文章:

  • pclinuxos系统详解
  • 应急响应靶机——WhereIS?
  • CRM和SCRM有什么区别
  • python实现usb热插拔检测(windows)
  • Android Framework
  • LWIP传输层协议笔记
  • Git 用法总结
  • 微信小程序原生swiper高度自适应图片,不同屏幕适配,正方形1:1等比例图片轮播
  • E+H流量计与Profibus DP主站转Modbus RTU/TCP网关通讯
  • DeepSeek新玩法: RAG Chatbot 3.0测试人的新大脑
  • 深入探讨dubbo组件的实践
  • Dapp开发-如何开发一个dapp
  • Vue 2 项目中配置 Tailwind CSS 和 Font Awesome 的最佳实践
  • MYSQL之表的约束
  • rbac模型详解
  • PHP编写图书信息爬虫程序
  • 力扣451:根据字符频率排序(桶排序)
  • 快解析为TPDDNS用户提供免费替换服务
  • 小白学习Java第18天(上):mybatis
  • 994. 腐烂的橘子
  • MYSQL时间函数、group by 和partition by的区别、组内编号leetcode学习
  • GitHub 趋势日报 (2025年05月11日)
  • LeetCode热题100——链表
  • docker-compose的yml文件配置deploy参数失效use the ‘deploy‘ key, which will be ignored.
  • MIMO 检测(2)--噪声白化
  • 雷池WAF的身份认证 - 钉钉配置教程
  • hi3516cv610的VPSS_ONLINE支持在vpss做图片放大的操作吗
  • IT团队如何通过ManageEngine卓豪Endpoint Central有效管理远程终端
  • 解决echartsV5+ restore后echarts显示空白
  • 防火墙来回路径不一致导致的业务异常