淺談Java父子類加載順序
先上桌結論,優先被加載的順序如下:
父類靜態成員變量 > 父類靜態代碼塊 > 子類靜態成員變量 > 子類靜態代碼塊 > 父類非靜態成員變量 > 父類非靜態代碼塊 > 父類構造方法 > 子類非靜態成員變量 > 子類非靜態代碼塊 > 子類構造方法
這麼長怎麼記呀?!
這裡幫大傢小結幾個特點:
- 成員變量 > 代碼塊 > 構造方法(構造器)。
- 靜態(共有) > 非靜態(私有)。
- 子類靜態 > 父類非靜態(私有)。
1. 靜態 > 非靜態
當且僅當該類在程序中第一次被 new(是第一次被類加載器調用時)才會觸發(不考慮永久代的回收),但隻調用 Main.class 是不會觸發的喲。
其實這也是 類優先於對象被加載 的體現。
2. 成員變量 > 成員方法 > 構造方法
- 可以這麼理解,加載這整個類,需要先知道類具有哪些成員變量,並且這些屬性初始化 (private String A = “”; 或者 private String A;) 完畢之後,這個類的對象才算是完整的。
- 其實 初始化非靜態成員變量 就是 new 對象的準備工作之一,等效於一個不接受任何類外部參數的構造方法。非靜態代碼塊也同理。因此,成員變量 > 非靜態代碼塊 > 構造方法。
3. 子類靜態 > 父類非靜態
結合第一條,也比較容易得出這一條規律。
測試代碼如下:
/** * @author Ander.Li */ 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("A non static"); } public A() { System.out.println("A init"); } } // B 是 A 的子類 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("B non static"); } 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("[First] new B:"); B b = new B(); System.out.println(); System.out.println("[Second] new B:"); b = new B(); } }
運行結果如下:
[First] new B:
Hi A
A static
Hi B
B static
Hi A2
A non static
A init
Hi B2
B non static
B init[Second] new B:
Hi A2
A non static
A init
Hi B2
B non static
B init
到此這篇關於Java父子類加載順序的實現的文章就介紹到這瞭,更多相關Java父子類加載順序內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!