Java代码块执行顺序实战:掌握Java初始化底层规律

作者:袖梨 2026-06-23
Java代码块执行顺序由JVM类加载与对象创建两阶段决定:静态代码块在类首次主动使用时执行一次,构造代码块每次new时在构造方法前执行,继承时严格遵循先父后子、先静态后实例的顺序。

Java代码块执行顺序不是靠记忆口诀,而是由JVM类加载与对象创建两个阶段共同决定的。核心在于分清“类初始化”和“对象初始化”——前者只发生一次,后者每次new都触发;两者各自有严格顺序,且受继承关系和声明位置双重约束。

静态代码块:类首次主动使用时才跑,且只跑一次

静态代码块(static {})属于类级别初始化,只有当类被首次主动使用时才会执行,比如调用静态方法、访问非final静态字段、new该类实例等。它和静态变量显式赋值一起被编译进方法,按源码从上到下顺序执行。

  • 父类静态部分一定先于子类执行,哪怕只写new Child()也会强制加载并初始化Parent
  • 编译期常量(如public static final int X = 1;)不触发初始化,直接内联到调用处
  • 若某个静态变量初始化时new了本类对象(如static A a = new A();),会提前进入对象初始化流程,但此时类尚未完成初始化,容易引发循环依赖或字段为默认值的问题

构造代码块:每次new都执行,优先于构造方法体

不带static的普通代码块({} )属于实例初始化,在每次调用构造器前自动运行,位置在实例变量显式赋值之后、构造方法体之前。它对所有构造器生效,适合提取共用初始化逻辑。

  • 多个构造代码块按源码声明顺序依次执行
  • 可访问实例变量、this、以及外部作用域的局部变量(若定义在方法内)
  • 它和实例变量直接赋值(如String name = "default";)处于同一层级,执行顺序完全取决于它们在类中出现的位置

继承场景下的完整流水线:先父后子、先静态后实例

创建子类对象时,JVM按固定链条推进,不可跳过、不可倒置:

立即学习“Java免费学习笔记(深入)”;

  • 父类静态变量默认值 → 父类静态变量显式赋值 + 静态块(从上到下)
  • 子类静态变量默认值 → 子类静态变量显式赋值 + 静态块(从上到下)
  • 父类实例变量默认值 → 父类实例变量显式赋值 + 构造块(从上到下)→ 父类构造方法体
  • 子类实例变量默认值 → 子类实例变量显式赋值 + 构造块(从上到下)→ 子类构造方法体

特别注意:父类构造器执行时,子类所有实例字段仍为默认值(0、null、false)。如果父类构造器中调用了被子类重写的方法,而该方法又访问了子类字段,就会读到未初始化状态。

普通代码块与方法内局部块:别混为一谈

写在类中、方法外的{}是实例初始化块,参与对象创建流程;而写在方法内部的{}只是局部作用域块,仅在方法执行到该位置时运行,与初始化无关。

  • 实例块只在new对象时执行一次,和你是否调用方法毫无关系
  • 方法内的{}不影响字段初始化,也不改变执行顺序,只用于限制变量生命周期或提升可读性
  • 验证技巧:用带标识的日志(如[I] init block[C] ctor)观察输出节奏,能清晰区分类加载、对象创建、方法调用三个阶段

相关文章

精彩推荐