瞭解包括繼承在內的初始化全過程,以對所發生的的一切有一個全局的把握,是頗有益的。java
請看下例:spa
package initialize; class Insect { private int i = 9; protected int j; Insect() { System.out.println("i = " + i + ", j = " + j); j = 39; } private static int x1 = printInit("static Insect.x1 initialized"); static int printInit(String s) { System.out.println(s); return 47; } } public class Beetle extends Insect{ private int k = printInit("Beelte.k initialized"); public Beetle(){ System.out.println("k = " + k); System.out.println("j = " + j); } private static int x2 = printInit("static Beetle.x2 initialized"); public static void main(String[] args) { System.out.println("Beetle constructor"); //Beetle beetle = new Beetle(); //此處先註釋 } }
運行結果:對象
在運行Beetle時,所發生的第一件事就是試圖訪問Beetel.main()方法(程序入口)(一個static方法),blog
因而加載器就開始啓動並找到Beetel類的編譯代碼(在Beetle.class文件中)。對他進行加載時,發現他有一個基類(這是由「extents」關鍵字得知的),因而繼續進行加載該基類。無論是否你是否打算生成一個該基類的對象,都會發生。從上出代碼Beelte.main()方法中的註釋代碼證實這一點。繼承
若是該基類還有其自身的基類,那麼第二個就會被加載,如此類推。接下來,根基類中的static方法初始化(此例中爲Insect)即會被執行,而後是下一個導出類,如此類推。it
至此,必要的類都加載完畢,對象就能夠建立了,咱們把上面的Beelte.main()方法中的註釋代碼取消,編譯
運行結果:class
首先對象中全部基本類型都會被設爲某默認值,對象引用都設爲null,而後基類的構造器會被調用。基類的構造器和導出類的構造器同樣,以相同的順序經歷相同的過程。基類構造器完成以後,實例變量按其次序被初始化。變量
習題:請證實加載類的動做只發生一次。證實該類的第一個實體的建立或static成員的訪問都有可能引發加載。引用
在Beelte.main()方法中添加一行代碼:
Beetle beetle2 = new Beetle();
輸出結果: