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

JVM 的类加载机制

JVM 的类加载机制是 Java 虚拟机动态加载、链接和初始化类的核心机制,它遵循严格的流程和规则,确保类的安全性和一致性。以下是详细说明:


类加载的流程

类加载分为 加载(Loading)→ 链接(Linking)→ 初始化(Initialization) 三个阶段:

  1. 加载(Loading)
    • 任务:查找并加载类的二进制字节流(如 .class 文件)。
    • 结果:在内存中生成一个代表该类的 Class 对象(方法区中)。
    • 数据来源:可以是本地文件、网络、JAR 包等。
  2. 链接(Linking)
    • 验证(Verification):检查字节码是否符合 JVM 规范(如魔数、语法合法性)。
    • 准备(Preparation):为类的静态变量分配内存并设置默认初始值(如 int 初始化为 0)。
    • 解析(Resolution):将常量池中的符号引用(如类名、方法名)转换为直接引用(内存地址)。
  3. 初始化(Initialization)
    • 执行类的 <clinit> 方法(编译器自动生成,包含静态变量赋值和静态代码块)。
    • 触发条件:首次主动使用类时(如 new 对象、访问静态变量/方法、反射调用等)。

双亲委派模型(Parent Delegation Model)

类加载器通过层级关系协作,确保核心类库的安全性,避免重复加载。

  1. 层级结构
    • Bootstrap ClassLoader(启动类加载器):
      加载 JAVA_HOME/lib 下的核心类库(如 rt.jar),由 C++ 实现,无父类。
    • Extension ClassLoader(扩展类加载器):
      加载 JAVA_HOME/lib/ext 目录的扩展类。
    • Application ClassLoader(应用类加载器):
      加载用户类路径(classpath)下的类,默认的类加载器。
    • 自定义 ClassLoader:用户可继承 ClassLoader 实现自定义加载逻辑。
  2. 工作流程

示例:加载用户自定义的 java.lang.String 类时,最终会由 Bootstrap ClassLoader 加载核心库的 String,避免用户篡改。

- 收到加载请求时,优先委派给父类加载器处理。  
- 若父类无法完成(在自己的搜索范围内找不到类),子类才会尝试加载。
  1. 优势
    • 避免重复加载,确保类全局唯一。
    • 保护核心类库不被自定义类覆盖。

打破双亲委派的场景

某些场景需要绕过双亲委派机制:

  1. SPI 服务加载(如 JDBC)
    Java 核心库(如 java.sql.Driver)由 Bootstrap ClassLoader 加载,而 SPI 实现类(如 MySQL 驱动)由应用类加载器加载。此时通过 线程上下文类加载器(Thread Context ClassLoader) 实现父类加载器请求子类加载器完成加载。
  2. 热部署/热加载
    如 Tomcat 为每个 Web 应用提供独立的类加载器,支持应用级类隔离和重新加载。
  3. 自定义类加载器
    用户可重写 loadClass() 方法改变委派逻辑。

类初始化的条件

类必须初始化的情况(主动引用):

  • new 实例对象、读写静态字段(非 final)、调用静态方法。
  • 反射调用(如 Class.forName("类名"))。
  • 初始化子类时,若父类未初始化,会触发父类初始化。

被动引用示例

class Parent {static int value = 10; static { System.out.println("Parent init!"); }
}
class Child extends Parent {static { System.out.println("Child init!"); }
}
// 访问 Parent.value 不会初始化 Child 类

类的卸载

  • 条件:类的 Class 对象无引用,且对应的类加载器被回收。
  • 实现:由 JVM 的垃圾回收机制完成,通常发生在方法区(元空间)内存不足时。

总结

JVM 类加载机制通过 双亲委派模型分阶段加载 确保类的安全加载与隔离,同时支持灵活扩展(如 SPI、热部署)。理解其原理有助于解决类冲突、实现动态加载等高级场景。

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

相关文章:

  • 进程管理(第二、三、四章)
  • 【车用永磁同步电机随机开关频率控制策略:高频谐波抑制的工程实践】
  • Python入门手册:条件判断
  • 云原生安全之网络IP协议:从基础到实践指南
  • mysql都有哪些锁?
  • 历年北京理工大学保研上机真题
  • 分布式缓存:ZSET → MGET 跨槽(cross‐slot)/ 并发 GET解决思路
  • 第十九章:数据治理之数据指标(一):数据指标工具之【指标口径管理系统】与【指标数据查询系统】
  • AnyIOasyncio 现代化方法
  • Ntfs!NtfsReadBootSector函数分析之nt!CcGetVacbMiss中得到一个nt!_VACB结构
  • 李宏毅《机器学习2025》笔记 第二讲 —— AI Agent
  • Dubbo与OpenFeign的区别
  • Apache 高级配置实战:从连接保持到日志分析的完整指南
  • 用python实现中国象棋
  • Tool-Star新突破!RL赋能LLM多工具协同推理,性能全面超越基线方法
  • Linux的进程控制
  • 基于RedisBloom的JWT黑名单管理方案
  • 【2025】ubuntu22.04 docker安装全过程
  • Odoo 前端开发框架技术全面解析
  • Odoo: Owl Props 深度解析技术指南
  • Linux操作系统之进程(三):进程优先级与进程切换调度
  • npm幻影依赖问题
  • npm修改镜像的教程,将npm镜像修改为国内地址增加下载速度
  • SpringBoot-11-基于注解和XML方式的SpringBoot应用场景对比
  • 【微服务】SpringBoot 对接飞书审批流程使用详解
  • [Excel VBA]如何製作買三送一優惠條件的POS結帳介面?
  • 论文阅读笔记——Janus,Janus Pro
  • java高级 -Junit单元测试
  • JVM八股速查
  • MySQL的索引