Java类加载过程全解析
时间:2026-01-28 19:54:43 228浏览 收藏
“纵有疾风来,人生不言弃”,这句话送给正在学习文章的朋友们,也希望在阅读本文《Java类加载过程详解》后,能够真的帮助到大家。我也会在后续的文章中,陆续更新文章相关的技术文章,有好的建议欢迎大家在评论留言,非常感谢!
类加载发生在运行时按需触发,如new实例化、调用静态方法、访问静态字段(非final)、Class.forName()、初始化子类等;被动引用不触发。

类加载发生在什么时候
Java 类加载不是在编译时完成的,而是在运行时按需触发。最常见的情况是首次主动使用某个类:比如 new 实例化、调用静态方法、访问静态字段(非 final 常量)、反射(Class.forName())、初始化子类(会先加载并初始化父类)等。被动引用(如子类引用父类静态字段、数组定义 MyClass[])不会触发加载。
注意:ClassLoader.loadClass(String name) 默认只执行加载和链接(验证、准备),不触发初始化;而 Class.forName(String name) 默认会初始化类 —— 这个差异常被忽略,导致静态块没执行、配置未加载。
加载、链接、初始化三阶段分别干啥
类加载过程严格分为三个阶段,不可跳过或逆序:
- 加载(Loading):通过类名定位字节码(从 classpath、jar、网络、动态生成等),读入二进制流,生成
java.lang.Class对象。此时类还没“活”起来,只是有了内存里的结构。 - 链接(Linking) 又分三步:
• 验证(Verification):检查字节码是否符合 JVM 规范(比如方法签名是否合法、栈是否溢出);
• 准备(Preparation):为类变量(static字段)分配内存并设默认值(如int设为 0,Object设为null),注意:不执行=后的赋值或静态代码块;
• 解析(Resolution):把符号引用(如类名、方法名字符串)转为直接引用(内存地址或句柄)。 - 初始化(Initialization):真正执行类构造器
方法,也就是静态变量赋值语句 + 静态代码块,按源码顺序执行。这是唯一允许用户代码介入的阶段。
双亲委派模型怎么打破又为何要打破
默认情况下,每个 ClassLoader 先委托父加载器尝试加载,直到启动类加载器(Bootstrap)。这种机制保证了核心类(如 java.lang.Object)不会被用户自定义版本替换,增强安全性。
但有些场景必须打破它:
- 热部署 / 模块隔离:OSGi、Tomcat 的 webapp 类加载器,需要各自独立加载相同类名的不同版本;
- 基础类依赖实现类:JDBC 的
DriverManager在rt.jar中,但它要加载用户提供的com.mysql.cj.jdbc.Driver,只能靠Thread.currentThread().getContextClassLoader()—— 这就是典型的“父加载器请求子加载器干活”; - 自定义加载逻辑:比如从加密 ZIP 或远程 HTTP 加载 class,需重写
findClass(),而非loadClass()(否则破坏委派)。
打破的关键是:重写 loadClass() 并去掉 super.loadClass() 调用,或直接调用 findClass();但务必小心,避免重复加载或类冲突(LinkageError)。
如何观察类加载过程
最直接的方式是加 JVM 参数:
-verbose:class
它会打印每一行 “[Loaded xxx from yyy]”,但信息太泛。更精准可配合:
-XX:+TraceClassLoading(同 -verbose:class)-XX:+TraceClassResolution:看符号引用解析细节- JDK9+ 推荐用
jcmd或 JFR(Java Flight Recorder)抓取类加载事件VM.native_memory summary - 代码中监听:注册
java.lang.instrument.ClassFileTransformer,或用ManagementFactory.getClassLoadingMXBean()查统计量
别依赖 IDE 的 “Debug Class Loading” 插件——它们往往只捕获部分事件,且干扰真实加载路径。真要调试,得在自定义 ClassLoader 的 defineClass() 和 resolveClass() 里打日志。
好了,本文到此结束,带大家了解了《Java类加载过程全解析》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
486 收藏
-
222 收藏
-
106 收藏
-
180 收藏
-
460 收藏
-
102 收藏
-
310 收藏
-
391 收藏
-
153 收藏
-
391 收藏
-
484 收藏
-
115 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习