這其實(shí)是去年校招時(shí)我遇到的一道阿里巴巴的筆試題(承認(rèn)有點(diǎn)久遠(yuǎn)了-。-),嗯,如果我沒(méi)記錯(cuò)的話,當(dāng)時(shí)是作為Java方向的一道選做大題。當(dāng)然題意沒(méi)有這么直白,題目只要求你寫出程序運(yùn)行后所有System.out.println的輸出結(jié)果,其中程序是題目給的,而各個(gè)System.out.println的執(zhí)行順序不同會(huì)導(dǎo)致最后程序輸出的結(jié)果也不同。
具體的題目我肯定記不清,不過(guò)我們可以換個(gè)直接的問(wèn)法,如果類A和類B中有靜態(tài)變量,靜態(tài)語(yǔ)句塊,非靜態(tài)變量,非靜態(tài)語(yǔ)句塊,構(gòu)造函數(shù),靜態(tài)方法,非靜態(tài)方法,同時(shí)類A繼承類B,請(qǐng)問(wèn)當(dāng)實(shí)例化A時(shí),類內(nèi)部的加載順序是什么?
當(dāng)時(shí)我也是一頭霧水,事后我就自己寫了一個(gè)小Demo,這才知道了類內(nèi)部的實(shí)際加載順,測(cè)試代碼如下:
Class B:
- public class B{
- //靜態(tài)變量
- static int i=1;
- //靜態(tài)語(yǔ)句塊
- static {
- System.out.println("Class B1:static blocks"+i);
- }
- //非靜態(tài)變量
- int j=1;
- //靜態(tài)語(yǔ)句塊
- static{
- i++;
- System.out.println("Class B2:static blocks"+i);
- }
- //構(gòu)造函數(shù)
- public B(){
- i++;
- j++;
- System.out.println("constructor B: "+"i="+i+",j="+j);
- }
- //非靜態(tài)語(yǔ)句塊
- {
- i++;
- j++;
- System.out.println("Class B:common blocks"+"i="+i+",j="+j);
- }
- //非靜態(tài)方法
- public void bDisplay(){
- i++;
- System.out.println("Class B:static void bDisplay(): "+"i="+i+",j="+j);
- return ;
- }
- //靜態(tài)方法
- public static void bTest(){
- i++;
- System.out.println("Class B:static void bTest(): "+"i="+i);
- return ;
- }
- }
Class A:
- public class A extends B{
- //靜態(tài)變量
- static int i=1;
- //靜態(tài)語(yǔ)句塊
- static {
- System.out.println("Class A1:static blocks"+i);
- }
- //非靜態(tài)變量
- int j=1;
- //靜態(tài)語(yǔ)句塊
- static{
- i++;
- System.out.println("Class A2:static blocks"+i);
- }
- //構(gòu)造函數(shù)
- public A(){
- super();
- i++;
- j++;
- System.out.println("constructor A: "+"i="+i+",j="+j);
- }
- //非靜態(tài)語(yǔ)句塊
- {
- i++;
- j++;
- System.out.println("Class A:common blocks"+"i="+i+",j="+j);
- }
- //非靜態(tài)方法
- public void aDisplay(){
- i++;
- System.out.println("Class A:static void aDisplay(): "+"i="+i+",j="+j);
- return ;
- }
- //靜態(tài)方法
- public static void aTest(){
- i++;
- System.out.println("Class A:static void aTest(): "+"i="+i);
- return ;
- }
- }
Class ClassLoading :
- public class ClassLoading {
- public static void main (String args[]) {
- A a=new A();
- a.aDisplay();
- }
- }
程序運(yùn)行結(jié)果如圖:
通過(guò)上述示圖,我們可以比較清晰的看出java類的整個(gè)加載過(guò)程。
1.若要加載類A,則先加載執(zhí)行其父類B(Object)的靜態(tài)變量以及靜態(tài)語(yǔ)句塊(執(zhí)行先后順序按排列的先后順序)。
2.然后再加載執(zhí)行類A的靜態(tài)變量以及靜態(tài)語(yǔ)句塊。(并且1、2步驟只會(huì)執(zhí)行1次)
3.若需實(shí)例化類A,則先調(diào)用其父類B的構(gòu)造函數(shù),并且在調(diào)用其父類B的構(gòu)造函數(shù)前,依次先調(diào)用父類B中的非靜態(tài)變量及非靜態(tài)語(yǔ)句塊.最后再調(diào)用父類B中的構(gòu)造函數(shù)初始化。
4.然后再依次調(diào)用類A中的非靜態(tài)變量及非靜態(tài)語(yǔ)句塊.最后調(diào)用A中的構(gòu)造函數(shù)初始化。( 并且3、4步驟可以重復(fù)執(zhí)行)
5.而對(duì)于靜態(tài)方法和非靜態(tài)方法都是被動(dòng)調(diào)用,即系統(tǒng)不會(huì)自動(dòng)調(diào)用執(zhí)行,所以用戶沒(méi)有調(diào)用時(shí)都不執(zhí)行,主要區(qū)別在于靜態(tài)方法可以直接用類名直接調(diào)用(實(shí)例化對(duì)象也可以),而非靜態(tài)方法只能先實(shí)例化對(duì)象后才能調(diào)用。
OK,今天就總結(jié)到這里了,如果有地方說(shuō)的不好或有錯(cuò)誤的地方,歡迎大家指出,定當(dāng)改正,謝謝。