Java类加载与初始化顺序是怎样的

类初始化顺序为:父类静态变量与代码块→子类静态变量与代码块→父类实例变量与代码块→父类构造函数→子类实例变量与代码块→子类构造函数,且静态内容仅加载一次。

Java类加载与初始化顺序是理解程序执行流程的关键部分。它决定了静态变量、实例变量、构造函数、代码块等何时被加载和执行。下面详细说明这一过程。

类加载过程

类加载由JVM完成,主要分为三个阶段:

  • 加载(Loading):通过类的全限定名获取其二进制字节流,生成Class对象。
  • 连接(Linking):包括验证、准备和解析。
    • 验证:确保字节码安全合规。
    • 准备:为类变量(static字段)分配内存并设置默认初始值(如0、null)。
    • 解析:将符号引用转为直接引用。
  • 初始化(Initialization):执行类构造器方法,真正给静态变量赋初值,并执行静态代码块。

类初始化触发条件

以下情况会触发类的初始化:

  • 创建类的实例(new关键字)。
  • 访问类的静态变量(非编译期常量)或静态方法。
  • 反射调用(如Class.forName)。
  • 初始化一个子类时,父类若未初始化则先初始化父类。
  • JVM启动时执行主类(包含main方法的类)。

类中各成分初始化顺序

当类被初始化时,内部成员按如下顺序执行:

  1. 父类静态变量和静态代码块,按书写顺序执行。
  2. 子类静态变量和静态代码块,按书写顺序执行。
  3. 父类实例变量和普通代码块,按书写顺序执行。
  4. 父类构造函数。
  5. 子类实例变量和普通代码块,按书写顺序执行。
  6. 子类构造函数。

注意:静态内容只在类第一次加载时执行一次;每次创建实例都会执行实例相关的代码块和构造函数。

示例说明

看一个例子帮助理解:

class Parent {
  static int parentStatic = printAndReturn("Parent static field", 1);
  {
    System.out.println("Parent instance block");
  }
  Parent() {
    System.out.println("Parent constructor");
  }
  static {
    System.out.println("Parent static block");
  }
}

class Child extends Parent {
  static int childStatic = printAndReturn("Child static field", 2);
  {
    System.out.println("Child instance block");
  }
  Child() {
    System.out.println("Child constructor");
  }
  static {
    System.out.println("Child static block");
  }
  static int printAndReturn(String msg, int value) {
    System.out.println(msg);
    return value;
  }
}

public class Test {
  public static void main(String[] args) {
    new Child();
  }
}

输出结果为:

Parent static field
Parent static block
Child static field
Child static block
Parent instance block
Parent constructor
Child instance block
Child constructor

这个顺序清晰地展示了从父类静态 → 子类静态 → 父类实例 → 父类构造 → 子类实例 → 子类构造的过程。

基本上就这些。掌握这个流程有助于避免静态资源误用、理解单例模式实现原理以及排查初始化异常问题。