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

Java类加载机制

Java 类加载机制详解

Java 类加载机制是 JVM(Java 虚拟机)的核心组成部分,负责将编译后的.class文件(字节码)从磁盘、网络或其他来源加载到内存中,并转换为可执行的 Java 类对象。整个过程遵循双亲委派模型,确保安全性和一致性。

1. 类加载的生命周期(Class Lifecycle)

一个类从加载到卸载,主要经历以下 5 个阶段:

  1. 加载(Loading)

    • 查找并读取字节码文件(.class),通过类加载器将字节码加载到 JVM 的方法区(JDK 8 前是永久代,JDK 8+ 是元空间)。
    • 生成对应的java.lang.Class对象(在堆中)。
  2. 连接(Linking)

    • 验证(Verification):确保字节码的安全性(格式、指令合法性等),防止恶意代码。
    • 准备(Preparation):为类的静态变量分配内存,并设置默认初始值(int=0, boolean=false, 引用=null 等)。
    • 解析(Resolution):将常量池中的符号引用转换为直接引用(可选,延迟解析)。
  3. 初始化(Initialization)

    • 执行类的静态初始化代码:静态变量赋值 + 静态代码块。
    • 初始化发生在类首次主动使用时(如 new、访问静态成员、反射等)。
    • <clinit>方法(编译器自动生成)。
  4. 使用(Using)

  5. 卸载(Unloading)

    • 当类不再被任何对象引用,且类加载器也被回收时,由 GC 卸载(自定义类加载器才可能被卸载)。
2. 类加载器(ClassLoader)

Java 默认使用三层类加载器(Bootstrap + Extension + Application),构成层次结构:

类加载器名称加载路径特点
Bootstrap ClassLoader(启动类加载器)$JAVA_HOME/lib中的核心类库(如 rt.jar:Object、String、System 等)用 C++ 实现,无 Java 对象引用(null)
Extension ClassLoader(扩展类加载器)$JAVA_HOME/lib/ext目录下的 jar父是 Bootstrap
Application ClassLoader(应用类加载器,也称 System ClassLoader)classpath 指定的路径(项目中的 classes、依赖 jar)父是 Extension,通常加载我们写的代码

此外还有:

  • 自定义类加载器:继承ClassLoader,重写findClass()方法。
  • 线程上下文类加载器(Thread Context ClassLoader):用于打破双亲委派(如 SPI 机制:JDBC、Servlet 容器)。
3. 双亲委派模型(Parental Delegation Model)

这是 Java 类加载机制的核心设计,工作流程:

  1. 当一个类加载器收到类加载请求时,不会自己先尝试加载
  2. 而是将请求向上委托给父类加载器(递归)。
  3. 只有当父类加载器无法找到该类时,当前类加载器才会尝试自己加载。

优点

  • 安全性:防止用户自定义恶意类覆盖核心类(如自定义 java.lang.String)。
  • 避免重复加载:同一个类只会被加载一次(由最高层的加载器加载)。
  • 命名空间隔离:不同类加载器加载的同名类是不同的(Class 对象不同)。

打破双亲委派的情况

  • 重写loadClass()方法(不推荐)。
  • SPI 机制(Service Provider Interface):如 JDBC DriverManager 使用线程上下文类加载器加载驱动。
  • 模块化系统(Java 9+ 的模块路径)。
  • 应用容器(如 Tomcat、Spring Boot)实现隔离。
4. 类加载的触发时机(首次主动使用)

以下操作会触发类初始化(执行<clinit>):

  • new创建对象
  • 访问静态变量或静态方法(包括赋值)
  • 反射:Class.forName("com.test.MyClass")(默认初始化,可传 false 跳过)
  • 初始化子类时,先初始化父类
  • 启动类(包含 main 方法的类)被加载时
  • 使用java.lang.invoke.MethodHandle的 REF_getStatic 等

不会触发初始化的情况

  • 访问常量(编译期常量,如 final static int A = 5)
  • 子类访问父类的静态字段(只初始化父类)
  • Class.forName("xxx", false, loader)
5. 示例代码演示
classParent{static{System.out.println("Parent static block");}publicstaticintvalue=100;}classChildextendsParent{static{System.out.println("Child static block");}}publicclassTest{publicstaticvoidmain(String[]args){System.out.println(Child.value);// 输出:Parent static block → 100// Child static block 不会执行,因为 Child 类未被主动使用}}
6. 常见面试题
  • 一个类会被加载几次?
    正常情况下只加载一次(由某个类加载器加载)。

  • 如何打破双亲委派?
    自定义类加载器,重写loadClass(),或使用线程上下文类加载器。

  • Tomcat 如何实现 Web 应用隔离?
    每个 Web 应用使用独立的 WebappClassLoader,优先加载本应用的类,打破双亲委派。

  • 为什么 String 类不能被自定义覆盖?
    因为它由 Bootstrap ClassLoader 加载,用户自定义的 String 在不同命名空间。

总结

Java 类加载机制的核心是:

  • 生命周期:加载 → 连接(验证+准备+解析) → 初始化 → 使用 → 卸载
  • 双亲委派模型:安全 + 避免重复加载
  • 类加载器层次:Bootstrap → Extension → Application → 自定义

理解类加载机制对深入掌握 JVM、Spring 容器、热部署、插件化开发至关重要。

http://www.jsqmd.com/news/205970/

相关文章:

  • 深度测评10个AI论文写作软件,继续教育学生轻松搞定论文!
  • AI不再“一本正经胡说八道“!LLM+RAG融合技术实战指南,让大模型回答有据可查,小白也能轻松上手
  • 2026 届计算机毕业设计全流程指南(从 0 到答辩)
  • js刷新页面的几种方法
  • TreeUtil树构建工具-超好用工具
  • 计算机专业毕设怎么选题?老师最容易通过的 20 个方向
  • Visual Studio 2022中配置cuda环境
  • AI 技术在英语培训中的应用
  • 从零到一全面掌握MySQL:安装配置、SQL详解与数据库实战理解
  • MySQL保姆级教程:从安装部署到核心概念,快速上手避坑指南
  • tcpdump抓包实战:命令行网络诊断利器
  • 飞书多维表格基础操作
  • 一篇搞定MySQL:从环境搭建到深入理解,高效入门数据库
  • 网络延迟与丢包问题排查实战
  • 欧盟EN 18031-1无线设备认证
  • MT-Safety 标签env 和 locale
  • EN 18031-1通用网络安全认证新规
  • 除了安全更新,EN 18031-1还有哪些重要的认证要求?
  • 写给开发者、内容创作者:当你“快做完了”却开始崩,这不是技术问题
  • 如何确保设备满足EN 18031-1标准中的安全更新要求?
  • 通达信专抓超跌副图无未来
  • 安达发|石油化工行业自动排产软件:驱动产业升级的核心引擎
  • 计算机毕设从选题到答辩,全程可指导(真实案例)
  • 通达信日周共振
  • AI 量化为什么不敢上线?——我的 Fail-Closed 模板实战
  • Python+Vue的大学生创新创业调查问卷系统 Pycharm django flask
  • 通达信回归斜率线
  • 红娘子双线强弱源码分享贴图
  • 如何在 ALCOR 强风控逻辑约束下,如何把 V8.2 年化拉到 28%?——一次“先别死,再赚钱”的量化实战复盘
  • 【毕业设计】机器学习基于python的砖头墙裂缝识别