加載順序
先上桌結論:
父類靜態屬性(成員變量) > 父類靜態代碼塊 > 子類靜態屬性 > 子類靜態代碼塊 > 父類非靜態屬性 > 父類非靜態代碼塊 > 父類構造器 > 子類非靜態屬性 > 子類非靜態代碼塊 > 子類構造器
這麼長怎麼記呀?!
這裏幫大家小結幾個特點:
- 靜態屬性和代碼塊,當且僅當該類在程序中第一次被 new 或者第一次被類加載器調用時纔會觸發(不考慮永久代的回收)。也正是因爲上述原因,類優先於對象 加載/new,即 靜態優先於非靜態。
- 屬性(成員變量)優先於構造方法,可以這麼理解,加載這整個類,需要先知道類具有哪些屬性,並且這些屬性初始化完畢之後,這個類的對象纔算是完整的。另外,非靜態代碼塊其實就是對象 new 的準備工作之一,算是一個不接受任何外來參數的構造方法。因此,屬性 > 非靜態代碼塊 > 構造方法。
- 有趣的是,靜態部分(前4個)是父類 > 子類,而 非靜態部分(後6個)也是父類 > 子類。
- 另外容易忽略的是,非靜態代碼塊在每次 new 對象時都會運行,可以理解:非靜態代碼塊是正式構造方法前的準備工作(非靜態代碼塊 > 構造方法)。
測試代碼如下:
/**
* @author Lean.Li
* @date 2018/10/15
*/
public class Main {
static class A {
static Hi hi = new Hi("A");
Hi hi2 = new Hi("A2");
static {
System.out.println("A static");
}
{
System.out.println("AAA");
}
public A() {
System.out.println("A init");
}
}
static class B extends A {
static Hi hi = new Hi("B");
Hi hi2 = new Hi("B2");
static {
System.out.println("B static");
}
{
System.out.println("BBB");
}
public B() {
System.out.println("B init");
}
}
static class Hi {
public Hi(String str) {
System.out.println("Hi " + str);
}
}
public static void main(String[] args) {
System.out.println("初次 new B:");
B b = new B();
System.out.println();
System.out.println("第二次 new B:");
b = new B();
}
}
運行結果如下:
初次 new B:
Hi A
A static
Hi B
B static
Hi A2
AAA
A init
Hi B2
BBB
B init
第二次 new B:
Hi A2
AAA
A init
Hi B2
BBB
B init