17370845950

Java类初始化顺序是如何决定的_Java类加载与初始化顺序解析
Java类初始化顺序为:先父类后子类、先静态后实例、先声明后构造;静态成员按源码顺序执行且只一次,实例成员在每次构造时按继承链和声明顺序初始化。

Java类的初始化顺序由类加载机制和语法规范共同决定,核心是“先父类后子类、先静态后实例、先声明后构造”。这个顺序不是凭空约定,而是JVM在类加载过程(加载→链接→初始化)中严格执行的规则,尤其在初始化阶段(方法执行时)体现得最明显。

静态成员的初始化优先于实例成员

所有static变量和static代码块按源码中出现的**从上到下顺序**执行,且只执行一次。它们属于类本身,不依赖对象创建。

  • 静态变量初始化表达式中若引用了尚未初始化的静态变量,其值为对应类型的默认值(如int为0,Objectnull
  • 多个static块之间也严格按书写顺序执行,可用于分段初始化逻辑
  • 父类的静态内容一定在子类的静态内容之前完成——因为子类初始化前,JVM必须确保其直接父类已完成初始化

父类初始化完成后再进行子类初始化

类初始化遵循继承链自顶向下:先触发Object,再逐级向下直到当前类。这个顺序体现在两个层面:

  • 静态部分:父类static变量 → 父类static块 → 子类static变量 → 子类static
  • 实例部分:每次创建对象时,先调用父类构造器(隐式或显式),再执行子类构造器;而构造器内部又会先执行父类实例变量赋值和实例块,再执行子类的

实例成员按声明顺序在构造器执行前初始化

static字段和实例初始化块({...}),在每次调用构造器时执行,顺序固定为:
父类实例变量/实例块 → 父类构造器主体 → 子类实例变量/实例块 → 子类构造器主体。

  • 即使子类构造器第一行写了super(...),父类的实例变量赋值和实例块仍会在super()调用返回后、父类构造器代码开始前完成
  • 实例块和字段初始化语句混写时,仍以源码顺序为准,与是否在构造器前后无关
  • 注意:字段声明中的初始化表达式(如private int x = getValue();)在此阶段求值,可能触发未预期的方法调用

类加载时机决定初始化何时发生

类初始化(即执行)不是类加载时立即发生,而是在**首次主动使用**该类时触发,例如:

  • 创建类的实例(new
  • 访问类的静态字段(被final修饰且在编译期能确定值的除外)
  • 调用类的静态方法
  • 反射调用(如Class.forName("X")
  • 初始化子类时发现父类未初始化,则先触发父类初始化

被动引用(如子类引用父类静态字段、数组定义、常量池符号引用)不会触发初始化。

基本上就这些。理解这个顺序的关键,是把“类加载”和“类初始化”区分开——加载只是把字节码读入内存,初始化才是执行static代码的真实时刻。只要记住“静态优先、父类先行、声明即序”,大部分初始化问题都能定位清楚。