这个问题其实看起来很简单,但是又很容易忘记和混淆。总结如下:
先看代码:
/**
* @ClassName: ClassInitializationDemo
* @description: 类初始化顺序
* @author: XZQ
* @create: 2020/3/1 9:34
**/
public class ClassInitializationDemo {
public static void main(String[] args) {
Father father = new Son();
System.out.println("====================");
Father father2 = new Son();
}
}
class Father {
public Father() {
System.out.println("加载父类构造函数");
}
public static String fatherStaicVar = "父类静态变量";
static {
System.out.println("加载父类静态代码块");
System.out.println("加载" + fatherStaicVar);
}
{
System.out.println("加载父类普通代码块");
}
}
class Son extends Father {
public Son() {
System.out.println("加载子类构造函数");
}
public static String sonStaicVar = "子类静态变量";
static {
System.out.println("加载子类静态代码块");
System.out.println("加载" + sonStaicVar);
}
{
System.out.println("加载子类普通代码块");
}
}
输出结果:
加载父类静态代码块
加载父类静态变量
加载子类静态代码块
加载子类静态变量
加载父类普通代码块
加载父类构造函数
加载子类普通代码块
加载子类构造函数
====================
加载父类普通代码块
加载父类构造函数
加载子类普通代码块
加载子类构造函数
总结
输出结果表明,程序的执行顺序为: 如果类还没有被加载:
1、先执行父类的静态代码块和静态变量初始化,并且静态代码块和静态变量的执行顺序只跟代码中出现的顺序有关。
2、执行子类的静态代码块和静态变量初始化。
3、执行父类的普通代码块和实例变量初始化
4、执行父类的构造函数
5、执行子类的普通代码块和实例变量初始化
6、执行子类的构造函数
另外如果类已经被加载:
静态代码块和静态变量就不用重复执行,再创建类对象时,只执行与实例相关的变量初始化和构造方法。
附:
当类加载器将类加载到JVM中的时候就会创建静态变量,这跟对象是否创建无关。静态变量加载的时候就会分配内存空间。静态代码块的代码只会在类第一次初始化的时候执行一次。一个类可以有多个静态代码块,它并不是类的成员,也没有返回值,并且不能直接调用。静态代码块不能包含this或者super,它们通常被用初始化静态变量。