JAVA設(shè)計(jì)模式之單例模式詳解
前言
在之前的文章里已經(jīng)介紹了設(shè)計(jì)模式以及設(shè)計(jì)原則的概念,接下來(lái)我們從單例模式入手深入學(xué)習(xí)幾種常用的JAVA設(shè)計(jì)模式,在實(shí)踐中加深理解。
一、單例模式是什么?
單例(Singleton)模式的定義:指一個(gè)類(lèi)只有一個(gè)實(shí)例,且該類(lèi)能自行創(chuàng)建這個(gè)實(shí)例的一種模式。
作用:單例模式是保證系統(tǒng)實(shí)例唯一性的重要手段。用于一個(gè)全局類(lèi)對(duì)象在多個(gè)地方被使用的場(chǎng)景下,保障了整個(gè)系統(tǒng)只有一個(gè)對(duì)象被使用,很好的節(jié)約了資源
實(shí)現(xiàn)方法:將類(lèi)的實(shí)例化方法私有化來(lái)防止程序通過(guò)其他方式創(chuàng)建該類(lèi)的實(shí)例,提供一個(gè)全局唯一獲取該類(lèi)實(shí)例的方法幫助用戶(hù)獲取類(lèi)的實(shí)例。
實(shí)現(xiàn)單例模式很簡(jiǎn)單,每次獲取前先判斷系統(tǒng)是否已經(jīng)存在單例對(duì)象,沒(méi)有就創(chuàng)建,有就返回這個(gè)對(duì)象。單例模式常見(jiàn)的寫(xiě)法有懶漢式單例和餓漢式單例。
二、懶漢式單例
定義:類(lèi)加載時(shí)沒(méi)有生成單例,第一次調(diào)用 getlnstance 方法時(shí)創(chuàng)建這個(gè)單例。
public class LazySingleton { //定義一個(gè)私有的靜態(tài)對(duì)象instance,靜態(tài)方法和屬性屬于類(lèi),能保障單例對(duì)象的唯一性 private static LazySingleton instance; //私有化構(gòu)造方法,只能在本類(lèi)中被訪問(wèn),其他類(lèi)中不能通過(guò)構(gòu)造方法直接創(chuàng)建對(duì)象 private LazySingleton() { } //提供一個(gè)全局唯一獲取實(shí)例的方法 public static synchronized LazySingleton getInstance(){ if(instance==null){ instance=new LazySingleton(); } return instance; } }
懶漢模式在獲取對(duì)象實(shí)例時(shí)做了加鎖操作,因此是線程安全的
測(cè)試代碼與結(jié)果
public class TestLazySingleton { public static void main(String[] args) { LazySingleton lazySingleton1=LazySingleton.getInstance(); LazySingleton lazySingleton2=LazySingleton.getInstance(); LazySingleton lazySingleton3=LazySingleton.getInstance(); System.out.println(lazySingleton1); System.out.println(lazySingleton2); System.out.println(lazySingleton3); } }
從上圖中可以看出雖然獲取了三次實(shí)例,但每次獲取的都是同一個(gè)實(shí)例,即一個(gè)類(lèi)只有一個(gè)實(shí)例。
三、餓漢式單例
定義:該模式的特點(diǎn)是類(lèi)一旦加載就創(chuàng)建一個(gè)單例,在調(diào)用 getInstance 方法之前單例已經(jīng)存在。
public class HungrySingleton { //類(lèi)加載完成后該類(lèi)的實(shí)例便已經(jīng)存在 private static HungrySingleton instance=new HungrySingleton(); //私有化構(gòu)造方法 private HungrySingleton(){ } //類(lèi)加載后實(shí)例就存在,不會(huì)出現(xiàn)線程安全問(wèn)題,不需要加鎖 public static HungrySingleton getInstance(){ return instance; } }
測(cè)試代碼和結(jié)果
public class TestHungrySingleton { public static void main(String[] args) { HungrySingleton hungrySingleton1=HungrySingleton.getInstance(); HungrySingleton hungrySingleton2=HungrySingleton.getInstance(); HungrySingleton hungrySingleton3=HungrySingleton.getInstance(); System.out.println(hungrySingleton1); System.out.println(hungrySingleton2); System.out.println(hungrySingleton3); } }
從上圖看出餓漢式單例在整個(gè)運(yùn)行過(guò)程中也只存在一個(gè)實(shí)例。
懶漢式單例和餓漢式單例的區(qū)別
1.懶漢模式在類(lèi)中定義了單例但是并未實(shí)例化,實(shí)例化是在方法中實(shí)現(xiàn)的,而餓漢模式定義的時(shí)候就進(jìn)行了實(shí)例化
2.懶漢模式需要在獲取實(shí)例的方法上加鎖保證線程安全,餓漢模式不需要加鎖。
四、雙重校驗(yàn)鎖
懶漢模式用到了synchronized,會(huì)導(dǎo)致很大的性能開(kāi)銷(xiāo),并且加鎖其實(shí)只需要在第一次初始化的時(shí)候用到,之后的調(diào)用都沒(méi)必要再進(jìn)行加鎖。
雙重校驗(yàn)鎖在懶漢模式的基礎(chǔ)上做了進(jìn)一步的優(yōu)化,給靜態(tài)對(duì)象加上volatile來(lái)保證有序性,第一次獲取對(duì)象時(shí)通過(guò)synchronize(Singleton.class)保障操作的唯一性。
public class LockSingleton { private volatile static LockSingleton lockSingleton; private LockSingleton(){} public static LockSingleton getInstance(){ if(lockSingleton==null){ synchronized (LockSingleton.class){ if (lockSingleton==null){ lockSingleton=new LockSingleton(); } } } return lockSingleton; } }
測(cè)試代碼與結(jié)果
public class LockTest { public static void main(String[] args) { LockSingleton lockSingleton1=LockSingleton.getInstance(); LockSingleton lockSingleton2=LockSingleton.getInstance(); LockSingleton lockSingleton3=LockSingleton.getInstance(); System.out.println(lockSingleton1); System.out.println(lockSingleton2); System.out.println(lockSingleton3); } }
執(zhí)行雙重檢查是因?yàn)椋绻鄠€(gè)線程同時(shí)了通過(guò)了第一次檢查,并且其中一個(gè)線程首先通過(guò)了第二次檢查并實(shí)例化了對(duì)象,那么剩余通過(guò)了第一次檢查的線程就不會(huì)再去實(shí)例化對(duì)象。
除了第一次創(chuàng)建實(shí)例的時(shí)候會(huì)出現(xiàn)加鎖的情況,后續(xù)的所有調(diào)用都會(huì)避免加鎖而直接返回,解決了性能消耗的問(wèn)題。
總結(jié)
關(guān)于單例模式本文介紹了懶漢模式、餓漢模式和雙重校驗(yàn)鎖。懶漢和餓漢模式的最大區(qū)別是定義實(shí)例的時(shí)候是否進(jìn)行實(shí)例化。雙重校驗(yàn)鎖是對(duì)懶漢模式的優(yōu)化,解決了性能消耗的問(wèn)題。
到此這篇關(guān)于JAVA設(shè)計(jì)模式之單例模式詳解的文章就介紹到這了,更多相關(guān)JAVA單例模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java實(shí)現(xiàn)數(shù)據(jù)庫(kù)連接池的方法
這篇文章主要介紹了Java實(shí)現(xiàn)數(shù)據(jù)庫(kù)連接池的方法,涉及java數(shù)據(jù)庫(kù)連接池的創(chuàng)建、連接、刷新、關(guān)閉及狀態(tài)獲取的常用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07教你怎么使用Java實(shí)現(xiàn)WebSocket
這篇文章主要介紹了教你怎么使用Java WebSocket,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們有很好的幫助,需要的朋友可以參考下2021-05-05深入理解Java對(duì)象的序列化與反序列化的應(yīng)用
本篇文章是對(duì)Java中對(duì)象的序列化與反序列化進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05關(guān)于java中@Async異步調(diào)用詳細(xì)解析附代碼
本文主要介紹了java關(guān)于@Async異步調(diào)用詳細(xì)解析附代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07java:try...catch跳過(guò)異常繼續(xù)處理循環(huán)問(wèn)題
這篇文章主要介紹了java:try...catch跳過(guò)異常繼續(xù)處理循環(huán)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10Spring Boot JPA中java 8 的應(yīng)用實(shí)例
這篇文章主要介紹了Spring Boot JPA中java 8 的應(yīng)用實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02