先看一個網上的例子:
public class SingleTon { PRivate static SingleTon singleTon = new SingleTon(); public static int count1; public static int count2 = 0; private SingleTon() { count1++; count2++; } public static void main(String[] args) { System.out.println(count1); System.out.println(count2); }}打?。?/p>10
沒有疑問,就不必往下看了,估計你已經懂了??赡芎芏嗳舜笠庵聲J為都是1,我就是其中之一。 我們簡單回顧下沒有繼承基類的簡單類的初始化過程: 類被加載->分配內存空間->靜態(tài)初始化->實例初始化 其中靜態(tài)初始化和實例初始化是有序自上而下的,看到這里應該已經明白了吧。 我們還是簡單說明白,類加載的時候基本類型count1和count2都被自動的初始化為0,然后執(zhí)行自上而下執(zhí)行靜態(tài)初始化,靜態(tài)初始化new SingleTon()最先執(zhí)行,2個值都變?yōu)榱?,然而count2又被重新賦值了,重新變?yōu)榱?。 我們來看下javap編譯后的匯編看看:
Constant pool://常量池 。。。省略。。。 #18 = Fieldref #1.#19 // reusing/SingleTon.count2:I 。。。省略。。。 Code: stack=2, locals=0, args_size=0 0: new #1 // class reusing/SingleTon 3: dup 4: invokespecial #13 // Method "<init>":()V //實例化在前 7: putstatic #16 // Field singleTon:Lreusing/SingleTon; 10: iconst_0 //常量0入棧 11: putstatic #18 // Field count2:I //為靜態(tài)域賦值 14: return }和我們分析得出的結果一模一樣。 如果是實例域,有將如何初始化,我們看代碼:
public class SingleTon { public int count1; public int count2 = 0; public SingleTon() { }}進行javap看看:
public class polymorphism.SingleTon { public int count1; public int count2; public polymorphism.SingleTon(); Code: 0: aload_0 1: invokespecial #11 // Method java/lang/Object."<init>":()V 4: aload_0 5: iconst_0 6: putfield #13 // Field count2:I //實例域count2,說明實例域和構造器共同完成實例初始化 9: return }通過上面的分析,我們知道,實例域初始化是類實例化的一部分,那么完整的類實例化順序應該如下:
新聞熱點
疑難解答