0

0

在Java中什么是类加载机制_JavaClassLoader工作原理解析

P粉602998670

P粉602998670

发布时间:2026-01-04 17:34:47

|

192人浏览过

|

来源于php中文网

原创

java类加载本质是将.class文件二进制数据转化为class对象的过程,分为加载、验证、准备、解析、初始化五阶段,遵循双亲委派模型以保障安全,且按需懒加载。

在java中什么是类加载机制_javaclassloader工作原理解析

Java 类加载机制的本质,是把磁盘上的 .class 文件二进制数据变成 JVM 能直接操作的 java.lang.Class 对象的过程;ClassLoader 就是干这件事的执行者——它不是“一次性加载全部类”,而是按需触发、分阶段转化、带安全校验的懒加载流水线。

类加载五阶段到底在做什么?

加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)这五个阶段顺序固定,但执行常交叉。关键不是背顺序,而是知道每步谁在动、动了什么:

  • 加载:由 ClassLoader 子类(如 AppClassLoader)完成,核心是调用 findClass(String name) 获取字节流,再用 defineClass(byte[] b) 注入 JVM 内存(方法区/元空间),最后在堆里 new 出一个 Class 对象;
  • 验证:JVM 自己做,检查魔数、版本号、符号引用合法性等,防止恶意字节码崩溃虚拟机
  • 准备:为 static 变量分配内存并设默认值(如 int 设为 0,不是你代码里写的 5);
  • 解析:把常量池里的符号引用(比如 "java/lang/Object")替换成内存中真实地址(直接引用),可能延迟到初始化后,以支持动态绑定;
  • 初始化:真正执行 <clinit>()</clinit> 方法——也就是静态变量赋值语句 + static{} 块,且 JVM 保证多线程下只执行一次。

双亲委派模型不是设计选择,而是安全刚需

当你调用 cl.loadClass("java.util.ArrayList"),实际走的是:AppClassLoader → ExtClassLoader → BootstrapClassLoader 逐级委托。这不是为了“优雅”,而是防止你写个假的 java.lang.String 替换掉真正的核心类:

  • Bootstrap 加载 rt.jar(JDK 8)或 modules(JDK 9+)里的 java.* 类,用 C++ 实现,没有 Java 父类;
  • ExtClassLoader 加载 $JAVA_HOME/lib/ext 下的扩展包(已逐步淘汰);
  • AppClassLoader(即 sun.misc.Launcher$AppClassLoader)加载 -cpCLASSPATH 下的应用类;
  • 自定义加载器必须显式调用 super(parent),否则会断掉委派链,容易引发 NoClassDefFoundErrorLinkageError

什么时候类真的被加载?别被“编译通过”骗了

类加载是“被动触发”的——哪怕你写了 new MyClass(),只要这行代码没被执行,MyClass 就不会加载。常见主动触发点:

立即学习Java免费学习笔记(深入)”;

AI Note
AI Note

AI Note 助手,像贴心女仆一样助力你的笔记!智能总结内容,精确划重点,提供专业建议,让学习与工作更高效。让你的笔记更清晰、有条理,知识尽在眼前!

下载
  • 执行 new 实例化(首次);
  • 访问非 final static 字段或调用静态方法(MyClass.fieldMyClass.method());
  • 反射调用 Class.forName("xxx")(注意:Class.forName("xxx", false, cl) 第二个参数为 false 时跳过初始化);
  • JVM 启动时加载含 main 方法的主类;
  • 子类初始化前,父类必须先初始化(但仅限“首次主动使用子类”时才触发父类初始化)。

陷阱示例:System.out.println(MyClass.CONSTANT) 不会加载 MyClass,因为 CONSTANTpublic static final 编译期常量,已被内联进调用方字节码。

自定义 ClassLoader 的典型场景与雷区

绕过双亲委派(重写 loadClass 而不调用 super.loadClass)只应在明确需要隔离或热替换时使用,比如插件系统、OSGi、JSP 容器或热部署工具

public class MyClassLoader extends ClassLoader {
    private final String baseDir;

    public MyClassLoader(String baseDir, ClassLoader parent) {
        super(parent); // 必须传 parent,否则无法访问系统类
        this.baseDir = baseDir;
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        byte[] bytes = loadClassBytes(name); // 自己读 .class 文件
        return defineClass(name, bytes, 0, bytes.length);
    }
}

容易踩的坑:

  • 忘记调用 super(parent) 导致 java.lang.* 类找不到;
  • 重复定义同一个类名(不同加载器)→ 生成不兼容的 Class 对象,强制转型会抛 ClassCastException
  • 未正确实现资源查找(getResource / getResources),导致 Properties配置文件、SPI 服务加载失败;
  • 类卸载困难:只有整个 ClassLoader 实例不可达,且其加载的所有类对象都无引用时,JVM 才可能回收元空间中的类元数据(JDK 8+)。

真正难的从来不是“怎么写一个加载器”,而是想清楚:这个类该不该由当前加载器负责?它依赖的其他类是否能被同一层级访问到?一旦打破委派,整个类可见性边界就变了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1565

2023.10.24

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1565

2023.10.24

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1565

2023.10.24

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1565

2023.10.24

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

608

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

334

2025.08.29

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

4

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号