摘要:這種情況帶來(lái)的后果是程序兩次創(chuàng)建了對(duì)象,這并不符合我們對(duì)單例模式的定義。實(shí)現(xiàn)的三個(gè)特性自由序列化線(xiàn)程安全保證單例。其次,有且僅有的構(gòu)造器,防止外部的額外構(gòu)造,這恰好與單例模式吻合。
單例模式保證一個(gè)類(lèi)僅有一個(gè)實(shí)例,并提供一個(gè)訪(fǎng)問(wèn)它的全局訪(fǎng)問(wèn)點(diǎn)。當(dāng)系統(tǒng)需要某個(gè)類(lèi)只能有一個(gè)實(shí)例時(shí),就可以采用單例模式。
保證單例模式僅有一個(gè)實(shí)例的核心思想是構(gòu)造方法私有化,即不允許外部調(diào)用該類(lèi)的構(gòu)造方法?;诖怂枷?,主要有以下兩種實(shí)現(xiàn)方式:
直接實(shí)例化直接實(shí)例化這種方式也稱(chēng)作“餓漢式”,它直接定義了靜態(tài)成員變量 s,并通過(guò) new Singleton() 完成了初始化,之后不再變化,是線(xiàn)程安全的。
這種方式也存在一定的資源浪費(fèi),當(dāng)沒(méi)有使用 Singleton 對(duì)象時(shí),程序依然會(huì)創(chuàng)建 Singleton 對(duì)象。
public class Singleton { private Singleton() {} private static final Singleton s = new Singleton(); public static Singleton getInstance() { return s; } }延遲實(shí)例化
既然直接實(shí)例化浪費(fèi)資源,那么我們是否可以考慮,在程序需要該對(duì)象的時(shí)候才創(chuàng)建它呢?當(dāng)然可以!
與直接實(shí)例化稍不同,單例成員變量 s 初始為 null,它在方法 getInstance() 內(nèi)部完成延遲實(shí)例化,并返回單例對(duì)象。
public class Singleton { private Singleton() {} private static Singleton s = null; public static Singleton getInstance() { if (s == null) { s = new Singleton(); } return s; } }
這種方式存在線(xiàn)程安全問(wèn)題。例如,假設(shè)兩個(gè)線(xiàn)程調(diào)用 getInstance() 方法,線(xiàn)程 1 執(zhí)行完 if(s == null),條件成立,在執(zhí)行實(shí)例化語(yǔ)句 s = new Singleton() 之前,線(xiàn)程 2 來(lái)了,此時(shí)線(xiàn)程 2 執(zhí)行 if(s == null),依然成立,進(jìn)入 if 語(yǔ)句體。這種情況帶來(lái)的后果是:程序兩次創(chuàng)建了對(duì)象,這并不符合我們對(duì)單例模式的定義。
針對(duì)這種情況,可以有以下四種解決方法:
完全同步完全同步方法,是在方法上加上 synchronized 同步。當(dāng)多線(xiàn)程同時(shí)訪(fǎng)問(wèn) getInstance() 方法時(shí),多線(xiàn)程是“串行”的。
public class Singleton { private Singleton() {} private static Singleton s = null; public static synchronized Singleton getInstance() { if (s == null) { s = new Singleton(); } return s; } }
這種方法,多線(xiàn)程每次訪(fǎng)問(wèn) getInstance() 都必須“串行”運(yùn)行,效率比較低。
部分同步部分同步方法通過(guò)雙重鎖部分同步機(jī)制獲得單例對(duì)象。因?yàn)榇a中有兩行相同的語(yǔ)句 if(s == null),故而叫做雙重鎖。第一個(gè) if 語(yǔ)句可并行,當(dāng)多線(xiàn)程均滿(mǎn)足該條件, synchronized 修飾的代碼必須串行運(yùn)行。這樣的話(huà),其實(shí)只需要在第一次創(chuàng)建對(duì)象(通過(guò)了第一個(gè) if 判斷)的時(shí)候進(jìn)行同步,效率較高。
public class Singleton { private Singleton() {} private volatile static Singleton s = null; public static Singleton getInstance() { if (s == null) { synchronized(Singleton.class) { if (s == null) { s = new Singleton(); } } } return s; } }
注意,volatile關(guān)鍵字是確保當(dāng) s 被初始化成 Singleton 實(shí)例時(shí),多個(gè)線(xiàn)程可以正確處理 s,即內(nèi)存可見(jiàn)性。
靜態(tài)內(nèi)部類(lèi)通過(guò)靜態(tài)內(nèi)部類(lèi) Inner 來(lái)實(shí)現(xiàn)單例對(duì)象。虛擬機(jī)加載應(yīng)用程序字節(jié)碼時(shí),單例對(duì)象并不會(huì)立即創(chuàng)建,當(dāng)?shù)谝淮芜\(yùn)行 Inner.s 時(shí),單例對(duì)象才動(dòng)態(tài)生成。這種實(shí)現(xiàn)方式無(wú) synchronized 關(guān)鍵字,提高了效率。
public class Singleton { private Singleton() {} private static class Inner { private static final Singleton s = new Singleton(); } public static Singleton getInstance() { return Inner.s; } }枚舉
這是單例模式的最佳實(shí)踐,它實(shí)現(xiàn)簡(jiǎn)單,并且在面對(duì)復(fù)雜的序列化或者反射攻擊的時(shí)候,能夠防止實(shí)例化多次。調(diào)用的時(shí)候只需要 Singleton.INSTANCE 即可。
public enum Singleton { INSTANCE; // var here public int var; // methods here public void otherMethods() { System.out.println("write other methods here..."); } }
enum 實(shí)現(xiàn) Singleton 的三個(gè)特性:自由序列化、線(xiàn)程安全、保證單例。
首先, enum 是由 class 實(shí)現(xiàn)的,它可以有 member 和 member function。另外,由于 enum 是通過(guò)繼承 Enum 類(lèi)實(shí)現(xiàn)的,enum 結(jié)構(gòu)不能作為子類(lèi)繼承其他類(lèi),但可以用來(lái)實(shí)現(xiàn)接口。此外 enum 類(lèi)不能被繼承,在反編譯中,可以發(fā)現(xiàn)該類(lèi)由 final 修飾。
其次,enum 有且僅有 private 的構(gòu)造器,防止外部的額外構(gòu)造,這恰好與單例模式吻合。
而對(duì)于序列化和反序列化,因?yàn)槊恳粋€(gè)枚舉類(lèi)型和枚舉變量在 JVM 中都是唯一的,即 Java在序列化和反序列化枚舉時(shí)做了特殊的規(guī)定,枚舉的 writeObject、readObject、readObjectNoData、writeReplace 和 readResolve 等方法是被編譯器禁用的,因此也不存在實(shí)現(xiàn)序列化接口后調(diào)用readObject 會(huì)破壞單例的問(wèn)題。
(完)
參考資料
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://www.ezyhdfw.cn/yun/71552.html
摘要:用來(lái)指向已創(chuàng)建好的實(shí)例構(gòu)造函數(shù)為空注意這里是關(guān)鍵這是我們需要調(diào)用的方法把函數(shù)也定義為空,這樣就大功告成啦。 接上一篇大話(huà)PHP設(shè)計(jì)模式之單例模式 這一篇介紹一下升級(jí)版的單例模式,廢話(huà)不說(shuō)先上代碼 不完美的單例模式 class singleMode { //用來(lái)指向已創(chuàng)建好的實(shí)例 public static $instance; //判斷是...
摘要:博主按每天一個(gè)設(shè)計(jì)模式旨在初步領(lǐng)會(huì)設(shè)計(jì)模式的精髓,目前采用靠這吃飯和純粹喜歡兩種語(yǔ)言實(shí)現(xiàn)。單例模式用途如果一個(gè)類(lèi)負(fù)責(zé)連接數(shù)據(jù)庫(kù)的線(xiàn)程池日志記錄邏輯等等,此時(shí)需要單例模式來(lái)保證對(duì)象不被重復(fù)創(chuàng)建,以達(dá)到降低開(kāi)銷(xiāo)的目的。 博主按:《每天一個(gè)設(shè)計(jì)模式》旨在初步領(lǐng)會(huì)設(shè)計(jì)模式的精髓,目前采用javascript(_靠這吃飯_)和python(_純粹喜歡_)兩種語(yǔ)言實(shí)現(xiàn)。誠(chéng)然,每種設(shè)計(jì)模式都有多種實(shí)...
摘要:博主按每天一個(gè)設(shè)計(jì)模式旨在初步領(lǐng)會(huì)設(shè)計(jì)模式的精髓,目前采用靠這吃飯和純粹喜歡兩種語(yǔ)言實(shí)現(xiàn)。單例模式用途如果一個(gè)類(lèi)負(fù)責(zé)連接數(shù)據(jù)庫(kù)的線(xiàn)程池日志記錄邏輯等等,此時(shí)需要單例模式來(lái)保證對(duì)象不被重復(fù)創(chuàng)建,以達(dá)到降低開(kāi)銷(xiāo)的目的。 博主按:《每天一個(gè)設(shè)計(jì)模式》旨在初步領(lǐng)會(huì)設(shè)計(jì)模式的精髓,目前采用javascript(_靠這吃飯_)和python(_純粹喜歡_)兩種語(yǔ)言實(shí)現(xiàn)。誠(chéng)然,每種設(shè)計(jì)模式都有多種實(shí)...
摘要:上面是簡(jiǎn)單的單例模式,自己寫(xiě)程序的話(huà)夠用了,如果想繼續(xù)延伸,請(qǐng)傳送至大話(huà)設(shè)計(jì)模式之單例模式升級(jí)版 看了那么多單例的介紹,都是上來(lái)就說(shuō)怎么做,也沒(méi)見(jiàn)說(shuō)為什么這么做的。那小的就來(lái)說(shuō)說(shuō)為什么會(huì)有單例這個(gè)模式以便更好的幫助初學(xué)者真正的理解這個(gè)設(shè)計(jì)模式,如果你是大神,也不妨看完指正一下O(∩_∩)O首先我不得不吐槽一下這個(gè)模式名字單例,初學(xué)者通過(guò)字面很難理解什么是單例,我覺(jué)得應(yīng)該叫唯一模式更貼切...
摘要:最近開(kāi)展了三次設(shè)計(jì)模式的公開(kāi)課,現(xiàn)在來(lái)總結(jié)一下設(shè)計(jì)模式在中的應(yīng)用,這是第一篇?jiǎng)?chuàng)建型模式之單例模式。不過(guò)因?yàn)椴恢С侄嗑€(xiàn)程所以不需要考慮這個(gè)問(wèn)題了。 最近開(kāi)展了三次設(shè)計(jì)模式的公開(kāi)課,現(xiàn)在來(lái)總結(jié)一下設(shè)計(jì)模式在PHP中的應(yīng)用,這是第一篇?jiǎng)?chuàng)建型模式之單例模式。 一、設(shè)計(jì)模式簡(jiǎn)介 首先我們來(lái)認(rèn)識(shí)一下什么是設(shè)計(jì)模式: 設(shè)計(jì)模式是一套被反復(fù)使用、容易被他人理解的、可靠的代碼設(shè)計(jì)經(jīng)驗(yàn)的總結(jié)。 設(shè)計(jì)模式不...
摘要:原文博客地址單例模式系統(tǒng)中被唯一使用,一個(gè)類(lèi)只有一個(gè)實(shí)例。中的單例模式利用閉包實(shí)現(xiàn)了私有變量?jī)烧呤欠裣嗟热躅?lèi)型,沒(méi)有私有方法,使用者還是可以直接一個(gè),也會(huì)有方法分割線(xiàn)不是單例最簡(jiǎn)單的單例模式,就是對(duì)象。 原文博客地址:https://finget.github.io/2018/11/06/single/ 單例模式 系統(tǒng)中被唯一使用,一個(gè)類(lèi)只有一個(gè)實(shí)例。實(shí)現(xiàn)方法一般是先判斷實(shí)例是否存在,...
閱讀 972·2019-08-30 15:54
閱讀 499·2019-08-30 12:51
閱讀 2098·2019-08-29 16:28
閱讀 2895·2019-08-29 16:10
閱讀 2391·2019-08-29 14:21
閱讀 474·2019-08-29 14:09
閱讀 2199·2019-08-23 16:13
閱讀 1287·2019-08-23 13:59