贝利信息

在Java里类加载过程是怎样的_Java类加载机制解析

日期:2026-01-22 00:00 / 作者:P粉602998670
类加载发生在运行时按需触发,如new实例化、调用静态方法、访问静态字段(非final)、Class.forName()、初始化子类等;被动引用不触发。

类加载发生在什么时候

Java 类加载不是在编译时完成的,而是在运行时按需触发。最常见的情况是首次主动使用某个类:比如 new 实例化、调用静态方法、访问静态字段(非 final 常量)、反射(Class.forName())、初始化子类(会先加载并初始化父类)等。被动引用(如子类引用父类静态字段、数组定义 MyClass[])不会触发加载。

注意:ClassLoader.loadClass(String name) 默认只执行加载和链接(验证、准备),不触发初始化;而 Class.forName(String name) 默认会初始化类 —— 这个差异常被忽略,导致静态块没执行、配置未加载。

加载、链接、初始化三阶段分别干啥

类加载过程严格分为三个阶段,不可跳过或逆序:

双亲委派模型怎么打破又为何要打破

默认情况下,每个 ClassLoader 先委托父加载器尝试加载,直到启动类加载器(Bootstrap)。这种机制保证了核心类(如 java.lang.Object)不会被用户自定义版本替换,增强安全性。

但有些场景必须打破它:

打破的关键是:重写 loadClass() 并去掉 super.loadClass() 调用,或直接调用 findClass();但务必小心,避免重复加载或类冲突(LinkageError)。

如何观察类加载过程

最直接的方式是加 JVM 参数:

-verbose:class

它会打印每一行 “[Loaded xxx from yyy]”,但信息太泛。更精准可配合:

别依赖 IDE 的 “Debug Class Loading” 插件——它们往往只捕获部分事件,且干扰真实加载路径。真要调试,得在自定义 ClassLoaderdefineClass()resolveClass() 里打日志。