Java类初始化顺序为:先父类后子类、先静态后实例、先声明后构造;静态成员按源码顺序执行且只一次,实例成员在每次构造时按继承链和声明顺序初始化。

Java类的初始化顺序由类加载机制和语法规范共同决定,核心是“先父类后子类、先静态后实例、先声明后构造”。这个顺序不是凭空约定,而是JVM在类加载过程(加载→链接→初始化)中严格执行的规则,尤其在初始化阶段(方法执行时)体现得最明显。
静态成员的初始化优先于实例成员
所有static变量和static代码块按源码中出现的**从上到下顺序**执行,且只执行一次。它们属于类本身,不依赖对象创建。
- 静态变量初始化表达式中若引用了尚未初始化的静态变量,其值为对应类型的默认值(如
int为0,Object为null) - 多个
static块之间也严格按书写顺序执行,可用于分段初始化逻辑 - 父类的静态内容一定在子类的静态内容之前完成——因为子类初始化前,JVM必须确保其直接父类已完成初始化
父类初始化完成后再进行子类初始化
类初始化遵循继承链自顶向下:先触发Object,再逐级向下直到当前类。这个顺序体现在两个层面:
-
静态部分:父类
static变量 → 父类static块 → 子类static变量 → 子类static块 - 实例部分:每次创建对象时,先调用父类构造器(隐式或显式),再执行子类构造器;而构造器内部又会先执行父类实例变量赋值和实例块,再执行子类的
实例成员按声明顺序在构造器执行前初始化
非static字段和实例初始化块({...}),在每次调用构造器时执行,顺序固定为:
父类实例变量/实例块 → 父类构造器主体 → 子类实例变量/实例块 → 子类构造器主体。
立即学习“Java免费学习笔记(深入)”;
- 即使子类构造器第一行写了
super(...),父类的实例变量赋值和实例块仍会在super()调用返回后、父类构造器代码开始前完成 - 实例块和字段初始化语句混写时,仍以源码顺序为准,与是否在构造器前后无关
- 注意:字段声明中的初始化表达式(如
private int x = getValue();)在此阶段求值,可能触发未预期的方法调用
类加载时机决定初始化何时发生
类初始化(即执行)不是类加载时立即发生,而是在**首次主动使用**该类时触发,例如:
- 创建类的实例(
new) - 访问类的静态字段(被
final修饰且在编译期能确定值的除外) - 调用类的静态方法
- 反射调用(如
Class.forName("X")) - 初始化子类时发现父类未初始化,则先触发父类初始化
被动引用(如子类引用父类静态字段、数组定义、常量池符号引用)不会触发初始化。
基本上就这些。理解这个顺序的关键,是把“类加载”和“类初始化”区分开——加载只是把字节码读入内存,初始化才是执行static代码的真实时刻。只要记住“静态优先、父类先行、声明即序”,大部分初始化问题都能定位清楚。










