深入解析Java的設(shè)計(jì)模式編程中單例模式的使用
定義:確保一個(gè)類只有一個(gè)實(shí)例,而且自行實(shí)例化并向整個(gè)系統(tǒng)提供這個(gè)實(shí)例。
類型:創(chuàng)建類模式
類圖:
類圖知識(shí)點(diǎn):
1.類圖分為三部分,依次是類名、屬性、方法
2.以<<開頭和以>>結(jié)尾的為注釋信息
3.修飾符+代表public,-代表private,#代表protected,什么都沒有代表包可見。
4.帶下劃線的屬性或方法代表是靜態(tài)的。
5.對(duì)類圖中對(duì)象的關(guān)系不熟悉的朋友可以參考文章:設(shè)計(jì)模式中類的關(guān)系。
單例模式應(yīng)該是23種設(shè)計(jì)模式中最簡(jiǎn)單的一種模式了。它有以下幾個(gè)要素:
- 私有的構(gòu)造方法
- 指向自己實(shí)例的私有靜態(tài)引用
- 以自己實(shí)例為返回值的靜態(tài)的公有的方法
來(lái)看一個(gè)簡(jiǎn)單的例子:
package com.wolf.action; import java.util.HashMap; import java.util.Map; public class demo { public static void main(String args[]) throws InstantiationException, IllegalAccessException, ClassNotFoundException { System.out.println(Son.getInstance().getName()); System.out.println("我是誰(shuí)"); } } class Son extends Father { private String name = "兒子"; final String CLASS = "demo"; protected String getName() { return this.query("aaa"); } public static Son getInstance() throws InstantiationException, IllegalAccessException, ClassNotFoundException { // 這里必須是全局路徑 否則無(wú)法找到 return (Son) instance("com.wolf.action.Son"); } } class Father { private static Map<String, Object> instance = new HashMap<String, Object>(); private String name = "父類"; protected void Fatcher() { System.out.println("我是父類"); } protected String query(String sql) { return sql + "has been done"; } public static Object instance(String objname) throws InstantiationException, IllegalAccessException, ClassNotFoundException { if (instance.get(objname) == null || !(instance.get(objname) instanceof Father)) { instance.put(objname, Class.forName(objname).newInstance()); } return instance.get(objname); } }
單例模式根據(jù)實(shí)例化對(duì)象時(shí)機(jī)的不同分為兩種:一種是餓漢式單例,一種是懶漢式單例。餓漢式單例在單例類被加載時(shí)候,就實(shí)例化一個(gè)對(duì)象交給自己的引用;而懶漢式在調(diào)用取得實(shí)例方法的時(shí)候才會(huì)實(shí)例化對(duì)象。代碼如下:
餓漢式單例
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è)對(duì)象,節(jié)省內(nèi)存空間。
- 避免頻繁的創(chuàng)建銷毀對(duì)象,可以提高性能。
- 避免對(duì)共享資源的多重占用。
- 可以全局訪問(wèn)。
適用場(chǎng)景:由于單例模式的以上優(yōu)點(diǎn),所以是編程中用的比較多的一種設(shè)計(jì)模式。我總結(jié)了一下我所知道的適合使用單例模式的場(chǎng)景:
- 需要頻繁實(shí)例化然后銷毀的對(duì)象。
- 創(chuàng)建對(duì)象時(shí)耗時(shí)過(guò)多或者耗資源過(guò)多,但又經(jīng)常用到的對(duì)象。
- 有狀態(tài)的工具類對(duì)象。
- 頻繁訪問(wèn)數(shù)據(jù)庫(kù)或文件的對(duì)象。
- 以及其他我沒用過(guò)的所有要求只有一個(gè)對(duì)象的場(chǎng)景。
單例模式注意事項(xiàng):
- 只能使用單例類提供的方法得到單例對(duì)象,不要使用反射,否則將會(huì)實(shí)例化一個(gè)新對(duì)象。
- 不要做斷開單例類對(duì)象與類中靜態(tài)引用的危險(xiǎn)操作。
- 多線程使用單例使用共享資源時(shí),注意線程安全問(wèn)題。
關(guān)于java中單例模式的一些爭(zhēng)議:
單例模式的對(duì)象長(zhǎng)時(shí)間不用會(huì)被jvm垃圾收集器收集嗎
看到不少資料中說(shuō):如果一個(gè)單例對(duì)象在內(nèi)存中長(zhǎng)久不用,會(huì)被jvm認(rèn)為是一個(gè)垃圾,在執(zhí)行垃圾收集的時(shí)候會(huì)被清理掉。對(duì)此這個(gè)說(shuō)法,筆者持懷疑態(tài)度,筆者本人的觀點(diǎn)是:在hotspot虛擬機(jī)1.6版本中,除非人為地?cái)嚅_單例中靜態(tài)引用到單例對(duì)象的聯(lián)接,否則jvm垃圾收集器是不會(huì)回收單例對(duì)象的。
對(duì)于這個(gè)爭(zhēng)議,筆者單獨(dú)寫了一篇文章進(jìn)行討論,如果您有不同的觀點(diǎn)或者有過(guò)這方面的經(jīng)歷請(qǐng)進(jìn)入文章單例模式討論篇:?jiǎn)卫J脚c垃圾收集參與討論。
在一個(gè)jvm中會(huì)出現(xiàn)多個(gè)單例嗎
在分布式系統(tǒng)、多個(gè)類加載器、以及序列化的的情況下,會(huì)產(chǎn)生多個(gè)單例,這一點(diǎn)是無(wú)庸置疑的。那么在同一個(gè)jvm中,會(huì)不會(huì)產(chǎn)生單例呢?使用單例提供的getInstance()方法只能得到同一個(gè)單例,除非是使用反射方式,將會(huì)得到新的單例。代碼如下
Class c = Class.forName(Singleton.class.getName()); Constructor ct = c.getDeclaredConstructor(); ct.setAccessible(true); Singleton singleton = (Singleton)ct.newInstance();
這樣,每次運(yùn)行都會(huì)產(chǎn)生新的單例對(duì)象。所以運(yùn)用單例模式時(shí),一定注意不要使用反射產(chǎn)生新的單例對(duì)象。
懶漢式單例線程安全嗎
主要是網(wǎng)上的一些說(shuō)法,懶漢式的單例模式是線程不安全的,即使是在實(shí)例化對(duì)象的方法上加synchronized關(guān)鍵字,也依然是危險(xiǎn)的,但是筆者經(jīng)過(guò)編碼測(cè)試,發(fā)現(xiàn)加synchronized關(guān)鍵字修飾后,雖然對(duì)性能有部分影響,但是卻是線程安全的,并不會(huì)產(chǎn)生實(shí)例化多個(gè)對(duì)象的情況。
單例模式只有餓漢式和懶漢式兩種嗎
餓漢式單例和懶漢式單例只是兩種比較主流和常用的單例模式方法,從理論上講,任何可以實(shí)現(xiàn)一個(gè)類只有一個(gè)實(shí)例的設(shè)計(jì)模式,都可以稱為單例模式。
單例類可以被繼承嗎
餓漢式單例和懶漢式單例由于構(gòu)造方法是private的,所以他們都是不可繼承的,但是其他很多單例模式是可以繼承的,例如登記式單例。
餓漢式單例好還是懶漢式單例好
在java中,餓漢式單例要優(yōu)于懶漢式單例。C++中則一般使用懶漢式單例。
單例模式比較簡(jiǎn)單,在此就不舉例代碼演示了。
- java設(shè)計(jì)模式之單例模式學(xué)習(xí)
- 淺析Java設(shè)計(jì)模式編程中的單例模式和簡(jiǎn)單工廠模式
- Java設(shè)計(jì)模式之單例模式實(shí)例詳解【懶漢式與餓漢式】
- Java設(shè)計(jì)模式之單例模式詳解
- java設(shè)計(jì)模式之單例模式的詳解及優(yōu)點(diǎn)
- Java設(shè)計(jì)模式之單例模式實(shí)例分析
- 簡(jiǎn)單講解在Java編程中實(shí)現(xiàn)設(shè)計(jì)模式中的單例模式結(jié)構(gòu)
- java 設(shè)計(jì)模式之單例模式
- Java設(shè)計(jì)模式系列之深入淺出單例模式
相關(guān)文章
java項(xiàng)目中的絕對(duì)路徑和相對(duì)路徑用法說(shuō)明
這篇文章主要介紹了java項(xiàng)目中的絕對(duì)路徑和相對(duì)路徑用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08springboot應(yīng)用訪問(wèn)zookeeper的流程
這篇文章主要介紹了springboot應(yīng)用訪問(wèn)zookeeper的流程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01Spring中的@PathVariable注解詳細(xì)解析
這篇文章主要介紹了Spring中的@PathVariable注解詳細(xì)解析,@PathVariable 是 Spring 框架中的一個(gè)注解,用于將 URL 中的變量綁定到方法的參數(shù)上,它通常用于處理 RESTful 風(fēng)格的請(qǐng)求,從 URL 中提取參數(shù)值,并將其傳遞給方法進(jìn)行處理,需要的朋友可以參考下2024-01-01SpringBoot調(diào)用第三方WebService接口的操作技巧(.wsdl與.asmx類型)
這篇文章主要介紹了SpringBoot調(diào)第三方WebService接口的操作代碼(.wsdl與.asmx類型 ),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08鴻蒙HarmonyOS App開發(fā)造輪子之自定義圓形圖片組件的實(shí)例代碼
這篇文章主要介紹了鴻蒙HarmonyOS App開發(fā)造輪子之自定義圓形圖片組件,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01Java構(gòu)造方法 super 及自定義異常throw合集詳解用法
異常是程序中的一些錯(cuò)誤,但不是所有錯(cuò)誤都是異常,且錯(cuò)誤有時(shí)候是可以避免的,super可以理解為是指向自己超(父)類對(duì)象的一個(gè)指針,而這個(gè)超類指的是離自己最近的一個(gè)父類,構(gòu)造器也叫構(gòu)造方法、構(gòu)造函數(shù),是一種特殊類型的方法,負(fù)責(zé)類中成員變量(域)的初始化2021-10-10將應(yīng)用程序進(jìn)行Spring6遷移的最佳使用方式
這篇文章主要介紹了將應(yīng)用程序進(jìn)行Spring6遷移的最佳方式,以及如何充分利用此升級(jí),需要的朋友可以參考下,如有錯(cuò)誤的地方還請(qǐng)指正2023-03-03Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(10)
下面小編就為大家?guī)?lái)一篇Java基礎(chǔ)的幾道練習(xí)題(分享)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧,希望可以幫到你2021-07-07