搜尋

首頁  >  問答  >  主體

類別載入機制 - java單例餓漢模式物件建立時間點疑問

關於java單例中餓漢式模式的解釋,大多如下:
餓漢模式線程安全的,在類創建的同時就已經創建好一個靜態的對象,相對與懶漢模式對象創建過早,浪費空間。

但是jvm中明確定義是:虛擬機規範則是嚴格規定了有且隻有5種情況必須立即對類進行“初始化”

1)遇到new、getstatic、putstatic或invokestatic這4條字節碼指令時,如果類沒有進行過初始化,則需要先觸發其初始化。
2)使用java.lang.reflect包的方法對類進行反射調用的時候,如果類沒有進行過初始化,則需要先觸發其初始化。
3)當初始化一個類的時候,如果發現其父類還沒有進行過初始化,則需要先觸發其父類的初始化。
4)當虛擬機啟動時,用戶需要指定一個要執行的主類(包含main()方法的那個類),虛擬機會先初始化這個主類。
5)當使用JDK 1.7的動態語言支持時,如果一個java.lang.invoke.MethodHandle實例最後的解析結果REF_get-Static、REF_putStatic、REF_invokeStatic的方法句柄,並且這個方法句柄所對應的類沒有進行過初始化,則需要先觸發其初始化。

public class Singleton {

private static Singleton instance = new Singleton();  
private Singleton (){}  
public static Singleton getInstance() {  
return instance;  
}  

}

根據第1)條可知道,隻有調用單例模式的getInstance的時候,才會初始化instance字段。

寫了個例子來驗證這點:

public class SingletonTest {

private static SingletonTest singleton=new SingletonTest();
private SingletonTest(){
    System.out.println("new signleton");
}
public static SingletonTest getSingleton(){
    return singleton;
}

}

public class Test {

public static void main(String args[]){
    SingletonTest singleton=null;
    System.out.println(singleton);
    singleton= SingletonTest.getSingleton();
}

}

執行main方法後的結果是:

null

new signleton

上麵的例子可以看出,在調用getInstance方法的時候,jvm才會對SingletonTest類初始化。

問題:在不使用反射模式加載單例類的情況下,懶漢模式和餓漢模式有區別嗎?

阿神阿神2835 天前753

全部回覆(1)我來回復

  • 巴扎黑

    巴扎黑2017-04-18 10:58:01

    .net 程式設計師一枚。
    你在SingletonTest 類別中加個靜態字段,不要呼叫getSingleton,直接呼叫這個字段,看看輸出什麼。

    還有不要死記單例模式的三種模式,一定要理解,然後才能靈活運用這三種模式。

    回覆
    0
  • 取消回覆