Java父子類加載順序

加載順序

先上桌結論:

父類靜態屬性(成員變量) > 父類靜態代碼塊 > 子類靜態屬性 > 子類靜態代碼塊 > 父類非靜態屬性 > 父類非靜態代碼塊 > 父類構造器 > 子類非靜態屬性 > 子類非靜態代碼塊 > 子類構造器

這麼長怎麼記呀?!

這裏幫大家小結幾個特點:

  1. 靜態屬性和代碼塊,當且僅當該類在程序中第一次被 new 或者第一次被類加載器調用時纔會觸發(不考慮永久代的回收)。也正是因爲上述原因,類優先於對象 加載/new,即 靜態優先於非靜態
  2. 屬性(成員變量)優先於構造方法,可以這麼理解,加載這整個類,需要先知道類具有哪些屬性,並且這些屬性初始化完畢之後,這個類的對象纔算是完整的。另外,非靜態代碼塊其實就是對象 new 的準備工作之一,算是一個不接受任何外來參數的構造方法。因此,屬性 > 非靜態代碼塊 > 構造方法
  3. 有趣的是,靜態部分(前4個)是父類 > 子類,而 非靜態部分(後6個)也是父類 > 子類
  4. 另外容易忽略的是,非靜態代碼塊在每次 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
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章