最近在学习jvm,看到其中讲到类初始化(initialization)时,看到内部类的加载过程,感到有些疑惑。
大家都知道,Java中,类的加载顺序是:
父类静态初始化块->子类静态初始化块->父类实例化初始化块->父类构造器->子类实例化初始化块->子类构造器。
可在以下这案例中,却有所不同
public class ClientTest {
static class Father {
public static int A = 1;
static {
A = 2;
}
}
static class Son extends Father {
public static int B = A;
}
public static void main(String[] args) {
System.out.println(Son.B);//输出2
}
}
而改变内部类变量的加载顺序后,却是这样的
static class Father {
static {
A = 2;
}
public static int A = 1;
}
static class Son extends Father {
public static int B = A;
}
public static void main(String[] args) {
System.out.println(Son.B);//输出1
}
之所以会出现这样的问题,是因为在类初始化中,还包括了静态属性的加载。静态构造块、静态类属性按出现在类定义里面的先后顺序初始化,同理非静态(实例化)的也是一样的,只是静态的只在加载字节码是执行一次,不管你new多少次,非静态(实例化)会在new多少次就执行多少次。
因此,上面Java的初始化顺序可扩展为:父类静态初始化块/静态属性(按顺序)->子类静态初始化块/静态属性->父类实例化初始化块/实例化属性(按顺序)->父类构造器->子类实例化初始化块/实例化属性->子类构造器。