類加載機制 - 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類初始化。
問題:在不使用反射模式加載單例類的情況下,懶漢模式和餓漢模式有區別嗎?
問題解答
回答1:.net 程序員一枚。你在SingletonTest 類中加個靜態字段,不調用getSingleton,直接調用這個字段,看看輸出什么。
還有不要死記單例模式的三種模式,一定要理解,然后才能靈活運用這三個模式。
相關文章:
1. 如何解決docker宿主機無法訪問容器中的服務?2. docker 下面創建的IMAGE 他們的 ID 一樣?這個是怎么回事????3. javascript - 請指條明路,angular的$event,在select中卻是undefined?4. 如何修改vim插件vimwiki中Vimwiki2HTML的一些細節5. javascript - JS事件委托問題6. javascript - 關于jquery的remove()方法7. JavaScript中怎么理解=、==和===8. java - Spring事務回滾問題9. javascript - vue.js如何遞歸渲染組件.10. mongodb - Mongo java驅動中,有沒有封裝好的函數,可以直接取到文檔中的某個具體的值,而不是一個文檔?或者有沒有方法實現??
