Java設(shè)計(jì)模式中的單例模式解析
Java單例模式
定義: 確保一個(gè)類只有一個(gè)實(shí)例,而且自行實(shí)例化并向整個(gè)系統(tǒng)提供這個(gè)實(shí)例。
類型: 創(chuàng)建類模式
類圖
類圖知識點(diǎn)
- 類圖分為三部分,依次是類名、屬性、方法
- 以<<開頭和以>>結(jié)尾的為注釋信息
- 修飾符+代表public,-代表private,#代表protected,什么都沒有代表包可見。
- 帶下劃線的屬性或方法代表是靜態(tài)的。
- 對類圖中對象的關(guān)系不熟悉的朋友可以參考文章:設(shè)計(jì)模式中類的關(guān)系。
單例模式應(yīng)該是23種設(shè)計(jì)模式中最簡單的一種模式了。
它有以下幾個(gè)要素:
- 私有的構(gòu)造方法
- 指向自己實(shí)例的私有靜態(tài)引用
- 以自己實(shí)例為返回值的靜態(tài)的公有的方法
單例模式根據(jù)實(shí)例化對象時(shí)機(jī)的不同分為兩種:一種是餓漢式單例,一種是懶漢式單例。
餓漢式單例在單例類被加載時(shí)候,就實(shí)例化一個(gè)對象交給自己的引用;
而懶漢式在調(diào)用取得實(shí)例方法的時(shí)候才會實(shí)例化對象。
代碼
餓漢式單例
public class Singleton { private static Singleton singleton = new Singleton(); private Singleton(){} public static Singleton getInstance(){ return singleton; } }
懶漢式單例
public class Singleton { private static Singleton singleton; private Singleton(){} public static synchronized Singleton getInstance(){ if(singleton==null){ singleton = new Singleton(); } return singleton; } }
單例模式的優(yōu)點(diǎn)
- 在內(nèi)存中只有一個(gè)對象,節(jié)省內(nèi)存空間。
- 避免頻繁的創(chuàng)建銷毀對象,可以提高性能。
- 避免對共享資源的多重占用。
- 可以全局訪問。
適用場景: 由于單例模式的以上優(yōu)點(diǎn),所以是編程中用的比較多的一種設(shè)計(jì)模式。我總結(jié)了一下我所知道的適合使用單例模式的場景:
- 需要頻繁實(shí)例化然后銷毀的對象。
- 創(chuàng)建對象時(shí)耗時(shí)過多或者耗資源過多,但又經(jīng)常用到的對象。
- 有狀態(tài)的工具類對象。
- 頻繁訪問數(shù)據(jù)庫或文件的對象。
- 以及其他我沒用過的所有要求只有一個(gè)對象的場景。
單例模式注意事項(xiàng)
- 只能使用單例類提供的方法得到單例對象,不要使用反射,否則將會實(shí)例化一個(gè)新對象。
- 不要做斷開單例類對象與類中靜態(tài)引用的危險(xiǎn)操作。
- 多線程使用單例使用共享資源時(shí),注意線程安全問題。
關(guān)于java中單例模式的一些爭議:
單例模式的對象長時(shí)間不用會被jvm垃圾收集器收集嗎
看到不少資料中說:如果一個(gè)單例對象在內(nèi)存中長久不用,會被jvm認(rèn)為是一個(gè)垃圾,在執(zhí)行垃圾收集的時(shí)候會被清理掉。
對此這個(gè)說法,筆者持懷疑態(tài)度,筆者本人的觀點(diǎn)是:在hotspot虛擬機(jī)1.6版本中,除非人為地?cái)嚅_單例中靜態(tài)引用到單例對象的聯(lián)接,否則jvm垃圾收集器是不會回收單例對象的。
在一個(gè)jvm中會出現(xiàn)多個(gè)單例嗎
在分布式系統(tǒng)、多個(gè)類加載器、以及序列化的的情況下,會產(chǎn)生多個(gè)單例,這一點(diǎn)是無庸置疑的。那么在同一個(gè)jvm中,會不會產(chǎn)生單例呢?使用單例提供的getInstance()方法只能得到同一個(gè)單例,除非是使用反射方式,將會得到新的單例。代碼如下
Class c = Class.forName(Singleton.class.getName()); Constructor ct = c.getDeclaredConstructor(); ct.setAccessible(true); Singleton singleton = (Singleton)ct.newInstance();
這樣,每次運(yùn)行都會產(chǎn)生新的單例對象。所以運(yùn)用單例模式時(shí),一定注意不要使用反射產(chǎn)生新的單例對象。
懶漢式單例線程安全嗎
主要是網(wǎng)上的一些說法,懶漢式的單例模式是線程不安全的,即使是在實(shí)例化對象的方法上加synchronized關(guān)鍵字,也依然是危險(xiǎn)的,但是筆者經(jīng)過編碼測試,發(fā)現(xiàn)加synchronized關(guān)鍵字修飾后,雖然對性能有部分影響,但是卻是線程安全的,并不會產(chǎn)生實(shí)例化多個(gè)對象的情況。
單例模式只有餓漢式和懶漢式兩種嗎
餓漢式單例和懶漢式單例只是兩種比較主流和常用的單例模式方法,從理論上講,任何可以實(shí)現(xiàn)一個(gè)類只有一個(gè)實(shí)例的設(shè)計(jì)模式,都可以稱為單例模式。
單例類可以被繼承嗎
餓漢式單例和懶漢式單例由于構(gòu)造方法是private的,所以他們都是不可繼承的,但是其他很多單例模式是可以繼承的,例如登記式單例。
餓漢式單例好還是懶漢式單例好
在java中,餓漢式單例要優(yōu)于懶漢式單例。C++中則一般使用懶漢式單例。 單例模式比較簡單,在此就不舉例代碼演示了。
到此這篇關(guān)于Java設(shè)計(jì)模式中的單例模式解析的文章就介紹到這了,更多相關(guān)Java單例模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)計(jì)算地理坐標(biāo)之間的距離
java實(shí)現(xiàn)計(jì)算地理坐標(biāo)之間的距離,主要是通過計(jì)算兩經(jīng)緯度點(diǎn)之間的距離來實(shí)現(xiàn),有需要的小伙伴參考下吧2015-03-03IDEA JarEditor編輯jar包方式(直接新增,修改,刪除jar包內(nèi)的class文件)
文章主要介紹了如何使用IDEA的JarEditor插件直接修改jar包內(nèi)的class文件,而不需要手動解壓、反編譯和重新打包,通過該插件,可以更方便地進(jìn)行jar包的修改和測試2025-01-01spring boot hutool整合email的詳細(xì)過程
這篇文章主要介紹了spring boot hutool整合email的相關(guān)知識,本文介紹兩種方式發(fā)送email文件,結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-03-03SpringMvc后臺接收json數(shù)據(jù)中文亂碼問題詳解
這篇文章主要介紹了SpringMvc后臺接收json數(shù)據(jù)中文亂碼問題詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09spring學(xué)習(xí)JdbcTemplate數(shù)據(jù)庫事務(wù)管理
這篇文章主要為大家介紹了spring學(xué)習(xí)JdbcTemplate數(shù)據(jù)庫事務(wù)管理,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05Java CompletableFuture 異步超時(shí)實(shí)現(xiàn)深入研究
這篇文章主要為大家介紹了Java CompletableFuture 異步超時(shí)實(shí)現(xiàn)深入研究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02