类加载发生在运行时按需触发,如new实例化、调用静态方法、访问静态字段(非final)、Class.forName()、初始化子类等;被动引用不触发。
Java 类加载不是在编译时完成的,而是在运行时按需触发。最常见的情况是首次主动使用某个类:比如 new 实例化、调用静态方法、访问静态字段(非 final 常量)、反射(Class.forName())、初始化子类(会先加载并初始化父类)等。被动引用(如子类引用父类静态字段、数组定义 MyClass[])不会触发加载。
注意:ClassLoader.loadClass(String name) 默认只执行加载和链接(验证、准备),不触发初始化;而 Class.forName(String name) 默认会初始化类 —— 这个差异常被忽略,导致静态块没执行、配置未加载。
类加载过程严格分为三个阶段,不可跳过或逆序:
java.lang.Class 对象。此时类还没“活”起来,只是有了内存里的结构。static 字段)分配内存并设默认值(如 int 设为 0,Object 设为 null),注意:不执行 = 后的赋值或静态代码块; 方法,也就是静态变量赋值语句 + 静态代码块,按源码顺序执行。这是唯一允许用户代码介入的阶段。默认情况下,每个 ClassLoader 先委托父加载器尝试加载,直到启动类加载器(Bootstrap)。这种机制保证了核心类(如 java.lang.Object)不会被用户自定义版本替换,增强安全性。
但有些场景必须打破它:

DriverManager 在 rt.jar 中,但它要加载用户提供的 com.mysql.cj.jdbc.Driver,只能靠 Thread.currentThread().getContextClassLoader() —— 这就是典型的“父加载器请求子加载器干活”;findClass(),而非 loadClass()(否则破坏委派)。打破的关键是:重写 loadClass() 并去掉 super.loadClass() 调用,或直接调用 findClass();但务必小心,避免重复加载或类冲突(LinkageError)。
最直接的方式是加 JVM 参数:
-verbose:class
它会打印每一行 “[Loaded xxx from yyy]”,但信息太泛。更精准可配合:
-XX:+TraceClassLoading(同 -verbose:class)-XX:+TraceClassResolution:看符号引用解析细节jcmd VM.native_memory summary 或 JFR(Java Flight Recorder)抓取类加载事件java.lang.instrument.ClassFileTransformer,或用 ManagementFactory.getClassLoadingMXBean() 查统计量别依赖 IDE 的 “Debug Class Loading” 插件——它们往往只捕获部分事件,且干扰真实加载路径。真要调试,得在自定义 ClassLoader 的 defineClass() 和 resolveClass() 里打日志。