Java类的初始化顺序与静态块的使用

静态成员按源码顺序初始化,先静态变量后静态块,父类优先于子类;实例创建时先父类字段和非静态块,再父类构造器,然后子类同理;静态块不可访问非静态成员,且失败会导致NoClassDefFoundError。

类加载时静态块和静态变量的执行顺序

静态成员(static 变量和 static 块)在类首次被主动使用时触发初始化,且只执行一次。它们按源码中从上到下的顺序执行,先初始化静态变量(含赋值表达式),再执行静态块——哪怕静态块写在变量声明之前,实际执行仍遵循声明位置顺序。

常见错误是误以为 static 块会“覆盖”或“延迟”静态变量初始化,其实二者同属类初始化阶段,共享同一执行流。

  • 如果静态变量初始化抛出异常(如 NullPointerException 在静态工厂方法中),后续静态块不会执行,且该类变为“初始化失败状态”,再次访问会直接抛 NoClassDefFoundError
  • 父类的静态成员总在子类之前完成初始化,这是 JVM 规范强制保证的
  • 静态变量的直接赋值(static int x = 5;)和静态块(static { ... })本质都是类初始化指令的一部分,字节码中统一为 方法

实例化时非静态块、构造器与字段初始化的顺序

每次 new 对象时,非静态初始化块和字段初始化语句按源码顺序执行,然后才调用构造器。注意:这发生在类已加载并完成静态初始化之后,且每创建一个实例都完整走一遍。

容易忽略的是,字段声明处的初始化(如 int y = getValue();)和非静态块({ ... })没有本质区别,JVM 把它们合并进每个构造器开头(在 super()this() 调用之后)。

  • 父类字段初始化 → 父类非静态块 → 父类构造器 → 子类字段初始化 → 子类非静态块 → 子类构造器
  • 若构造器第一行是 this(...),则当前类的字段初始化和非静态块会在目标构造器中执行,而非重复执行两次
  • 字段初始化中调用的方法如果是 override 的,会触发多态——但此时子类字段可能还未初始化(值为默认值),造成逻辑错误

静态块里能做什么?哪些操作要特别小心

static 块适合做一次性资源准备,比如加载配置、注册驱动、初始化缓存。但它不是万能的,很多看似合理的行为在类加载期会出问题。

  • 不能访问非静态成员(编译报错:non-static variable xxx cannot be referenced from a static context
  • 避免在静态块中调用可能触发其他类加载的方法(如反射 Class.forName("xxx")),易引发死锁或循环依赖
  • 不要在静态块中执行耗时 I/O(如读文件、连数据库)——它会阻塞所有对该类的首次访问,且无重试机制
  • 若需线程安全的单例初始化,优先用 static 字段 + final(如 private static final Singleton INSTANCE = new Singleton();),比静态块更简洁可靠

验证初始化顺序的最小可测代码

下面这段代码能清晰暴露 JVM 执行路径,建议复制到 IDE 中运行并观察输出:

class Parent {
    static { System.out.println("Parent static block"); }
    { System.out.println("Parent init block"); }
    Parent() { System.out.println("Parent constructor"); }
}

class Child extends Parent {
    static { System.out.println("Child static block"); }
    { System.out.println("Child init block"); }
    Child() { System.out.println("Child constructor"); }
}

public class InitOrder {
    public static void main(String[] args) {
        System.out.println("main start");
        new Child();
        new Child();
    }
}

输出顺序固定为:Parent static blockChild static blockmain startParent init blockParent constructorChild init blockChild constructor →(第二次 new 时跳过所有 static 块)→ Parent init block → ……

真正复杂的地方在于:静态初始化失败不可恢复,而实例初始化失败只影响当次对象创建;另外,接口中的 static 块(Java 8+)和 default 方法不参与类初始化流程,这点常被混淆。