类加载机制按需加载类并初始化,包括加载、验证、准备、解析和初始化五个阶段;JVM通过双亲委派模型确保安全性,由启动类加载器、扩展类加载器和应用程序类加载器协同工作;类在首次主动使用时初始化,如创建实例、访问静态变量、调用静态方法等;初始化顺序为:父类静态变量与静态代码块(按书写顺序)、子类静态变量与静态代码块、父类实例变量与非静态代码块、父类构造函数、子类实例变量与非静态代码块、子类构造函数;示例中new Child()触发Parent和Child的静态块依次执行,且仅执行一次。

Java中的类加载机制与类初始化顺序是理解程序运行时行为的关键。当一个类被使用时,JVM并不会立即加载所有相关的类,而是按需加载、链接并初始化,这个过程由类加载器和虚拟机协同完成。
类加载的核心阶段
类的生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using)和卸载(Unloading)。其中前五个阶段构成了类加载过程。
加载:通过类的全限定名获取其二进制字节流,并将其转化为方法区中的运行时数据结构,在内存中生成一个java.lang.Class对象作为入口。
验证:确保Class文件字节流符合当前虚拟机要求,防止恶意代码危害JVM安全。
立即学习“Java免费学习笔记(深入)”;
准备:为类的静态变量分配内存并设置默认初始值(如0、null等),不执行任何Java代码。
解析:将常量池内的符号引用替换为直接引用,比如把类名转成内存地址指针。
初始化:真正执行类中定义的Java代码,包括静态变量赋值语句和静态代码块,按源码顺序执行。
类加载器的层次结构
JVM采用双亲委派模型来组织类加载器,主要包括以下三种:
- 启动类加载器(Bootstrap ClassLoader):负责加载$JAVA_HOME/jre/lib下的核心类库,如rt.jar,用C++实现。
- 扩展类加载器(Extension ClassLoader):加载jre/lib/ext目录下的扩展jar包。
- 应用程序类加载器(Application ClassLoader):也叫系统类加载器,加载用户类路径(classpath)上的类。
当一个类加载请求到来时,先委托父加载器尝试加载,只有在父加载器无法完成时才由自己加载,这种机制保证了类的统一性和安全性。
触发类初始化的时机
类的初始化发生在首次主动使用时,以下是常见的五种情况:
- 创建类的实例,例如new一个对象。
- 访问或修改类的静态变量(非final字段)。
- 调用类的静态方法。
- 反射操作该类,比如
Class.forName()。 - 初始化一个子类时,若其父类还未初始化,则先初始化父类。
注意:访问static final基本类型或字符串常量不会触发初始化,因为它们在编译期就已嵌入常量池。
类初始化顺序详解
在一个类被初始化时,执行顺序严格遵循以下规则:
- 父类静态变量和静态代码块,按书写顺序执行。
- 子类静态变量和静态代码块,同样按出现顺序执行。
- 父类实例变量和非静态代码块,构造函数之前执行。
- 父类构造函数。
- 子类实例变量和非静态代码块。
- 子类构造函数。
示例说明:
class Parent {static int a = 1;
static { System.out.println("Parent static block, a=" + a); }
}
class Child extends Parent {
static int b = 2;
static { System.out.println("Child static block, b=" + b); }
}
public class Test {
public static void main(String[] args) {
new Child(); // 输出:Parent static block..., Child static block...
}
}
这里会先初始化Parent再初始化Child,静态部分只执行一次,后续创建更多Child实例不会再触发静态块。
基本上就这些。掌握类加载机制有助于排查ClassNotFoundException、NoClassDefFoundError等问题,也能更好理解框架底层原理,比如Spring如何管理Bean类的加载时机。










