成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一個單例還能寫出花來嗎?

開發(fā) 前端
單例可以說是最簡單的一個設計模式了,單例模式要求只能創(chuàng)建一個對象實例。通常的寫法是聲明私有的構(gòu)造函數(shù),提供靜態(tài)方法獲取單例的對象實例。

[[393262]]

 單例可以說是最簡單的一個設計模式了,單例模式要求只能創(chuàng)建一個對象實例。通常的寫法是聲明私有的構(gòu)造函數(shù),提供靜態(tài)方法獲取單例的對象實例。

常見的單例寫法就是餓漢式、懶漢式、雙重加鎖驗證、靜態(tài)內(nèi)部類和枚舉的方式,寫法可能大家都知道,不過針對不同的寫法還是有可以繼續(xù)深挖一下的地方,讓我們從最簡單的幾種寫法開始回顧單例,不想看前面的話直接往后翻好了。

回顧幾種實現(xiàn)方式

餓漢式

餓漢式的寫法通常靜態(tài)成員變量已經(jīng)是初始化好的,優(yōu)點是可以不加鎖就獲取到對象實例,線程安全,主要的缺點在于不是延加載,稍微存在內(nèi)存的浪費,因為如果初始化的邏輯較為復雜,比如存在網(wǎng)絡請求或者一些復雜的邏輯在內(nèi),就會產(chǎn)生內(nèi)存的浪費。

懶漢式

懶漢式的寫法解決了餓漢式浪費內(nèi)存的問題,在真正需要獲取實例對象的才去執(zhí)行初始化。

通常一般來說可能會有兩種方式,第一種就是不加鎖的寫法,很顯然這樣是肯定不行的,正常的方式一般都是通過同步鎖的方式加鎖獲取實例對象。

但是這種實現(xiàn)方式在之前的JDK版本synchronized沒有鎖優(yōu)化的情況每次獲取單例對象性能存在很大的問題,于是乎有了DCL的寫法。

雙重加鎖驗證DCL

于是為了解決懶漢式性能的問題,雙重加鎖驗證的寫法誕生了,先判斷一次空,真的為空再執(zhí)行加鎖,然后再判斷一次。

這樣的話,只有在實例對象是空的情況才會去加鎖創(chuàng)建對象,性能問題得到了一定程度上的解決,也不會和餓漢一樣有內(nèi)存浪費的問題。

但是,這個寫法也存在問題,就是會拿到未初始化完全的對象,我之前的一篇文章中也提到這個方式的問題,具體請看一次群聊引發(fā)的血案。

讓我這里復用一下我寫過的東西。

  1. 從CPU的角度來看,instance = new Instance()可以分為分為幾個步驟:
  2. 分配對象內(nèi)存空間
  3. 執(zhí)行構(gòu)造方法,對象初始化

instance指向分配的內(nèi)存地址

實際上,由于指令重排的問題,2、3的步驟可能會發(fā)生重排序,那么問題就發(fā)生了。

instance先被指向內(nèi)存地址,然后再執(zhí)行初始化,如果此時另外一個線程來訪問getInstance方法,就會拿到instance不是null,最后拿到的將是一個沒有被完全初始化的對象!

現(xiàn)在也有很多人說這個問題在高版本的JDK中已經(jīng)解決了,但是我是沒發(fā)現(xiàn)有什么直接證據(jù),如果你知道,請你告訴我。

靜態(tài)內(nèi)部類

這個通過JVM來保證創(chuàng)建單例對象的線程安全和唯一性,是比較好的辦法。

Singleton類加載的時候,SingletonHolder不會加載,只有在調(diào)用getInstance方法的時候才會執(zhí)行初始化,這樣既起到了懶加載的作用,同時又使用到了JVM類加載機制,保證了單例對象初始化的線程安全。

這種方式也是目前比較推薦的一種方式。

枚舉

通過枚舉來實現(xiàn)單例是Effective Java作者 Josh Bloch 提倡的方式,也是單例模式的最佳實現(xiàn)方式。

為了看清楚枚舉怎么實現(xiàn)單例模式的,我們來編譯一下枚舉生成的最終字節(jié)碼。

執(zhí)行javac Singleton.java生成class文件,接著執(zhí)行javap -p Singleton.class,得到如下內(nèi)容:

為了看到更詳細的內(nèi)容,我們執(zhí)行 javap -c Singleton。

通過最終生成的字節(jié)碼,我們其實發(fā)現(xiàn)本質(zhì)上枚舉的初始化通過static代碼塊來進行初始化。

考慮下類加載的幾個步驟,加載->驗證->準備->解析->初始化,最終初始化就是執(zhí)行static代碼塊,而static代碼塊是絕對線程安全的,只能由JVM來調(diào)度,這樣保證了線程安全。

枚舉的實現(xiàn)方式好處還不止于此,除了一目了然的實現(xiàn)簡單之外,還能防止其他幾種實現(xiàn)方式避免不了的幾個問題。

再說幾種方式的問題

反射破壞單例

除了枚舉之外,其他的幾種方式都可以通過反射的方式達到破壞單例的目的,就隨便以一個實現(xiàn)方式來舉例,這里最終的輸出結(jié)果是false。

如果拿去嘗試反射創(chuàng)建枚舉對象的話,則是會報錯,可以自己動手嘗試一下。

為什么會報錯,可以直接看一下newInstance的源碼,有一段特殊的關(guān)于枚舉類型的判斷,下圖中我紅色標記的部分。

序列化

除了眾所周知的使用反射來破壞單例之外,還有另外一種能破壞單例的方式就是序列化。

對上面的餓漢方法實現(xiàn)序列化,然后得到的結(jié)果是false,序列化前后對象發(fā)生了改變。

其實關(guān)鍵的部分在于ois.readObject方法,一路跟蹤最后找到一段代碼如下:

所以很明顯我們發(fā)現(xiàn)了最終實際上這里通過反射創(chuàng)建了一個新的對象,isInstantiable實際代表的應該是類或者屬性是序列化的,那么久就返回true,我們這里肯定是true,所以最終產(chǎn)生了一個新的對象。

枚舉為啥可以防止這個問題?枚舉的實現(xiàn)方式不太一樣而已,同樣跟蹤到枚舉部分的實現(xiàn)邏輯。

下圖中紅框標注的部分就是枚舉類型去實現(xiàn)反序列化的邏輯,最終只是通過valueOf方法查找枚舉,不存在新建一個對象的邏輯。

那么,怎么防止其他方式序列化對單例的破壞?再往下看看源碼,紅框標注的意思只要有readResolve方法就可以解決問題了。

實際上,最終解決方案也很簡單,單例類加上方法即可。

好了,打完收工。現(xiàn)在是北京時間4月15日凌晨1點整,困了,睡覺。

 

責任編輯:武曉燕 來源: 艾小仙
相關(guān)推薦

2024-02-22 10:02:03

單例模式系統(tǒng)代碼

2024-02-04 16:14:38

線程開發(fā)

2025-06-26 00:40:13

2021-10-11 08:08:02

Python異常程序

2022-09-29 08:39:37

架構(gòu)

2022-06-27 19:19:26

算法題青蛙跳臺階

2020-06-02 09:22:45

腳本CPUDDG

2025-02-07 09:34:12

2023-04-12 08:16:35

2021-08-04 12:26:00

Postman工具頻率

2017-09-18 09:03:36

線程安全單例

2018-04-20 14:35:18

電費電腦收益

2022-05-30 08:02:51

事務日志MySQL數(shù)據(jù)庫

2022-02-21 08:00:23

開發(fā)代碼程序員

2010-08-16 10:10:22

SQL腳本

2017-09-18 09:17:07

線程安全單例

2024-05-16 12:03:54

Python代碼開發(fā)

2023-04-10 08:11:23

@Lazy懶漢模式餓漢模式

2021-09-07 10:44:35

異步單例模式

2023-10-08 10:14:12

點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 中文字幕不卡在线88 | 久操国产| 国产黄色电影 | 久久er99热精品一区二区 | 五月天天丁香婷婷在线中 | 丁香综合 | 久久国内精品 | 日韩精品一区二区三区在线观看 | 久久不射电影网 | 欧美精品在欧美一区二区少妇 | 综合久久国产 | 欧美国产精品久久久 | 天天看天天摸天天操 | 欧州一区二区三区 | 中文字幕在线观看第一页 | 久草青青 | 波波电影院一区二区三区 | 91精品国产综合久久婷婷香蕉 | 精品视频在线观看 | 欧美日韩久久 | 成人精品一区二区 | wwwxxx日本在线观看 | zzzwww在线看片免费 | 婷婷综合激情 | 久久久久久av | 免费国产网站 | 91色视频在线观看 | 国产精品爱久久久久久久 | 99tv成人影院 | 国产精品色 | 亚洲一区二区三 | 亚洲欧美精品 | a在线视频 | 欧产日产国产精品国产 | 亚洲视频免费观看 | 久久久久一区 | 人人做人人澡人人爽欧美 | 四虎在线播放 | 免费看欧美一级片 | 不卡一区二区三区四区 | 国产精品成人一区二区三区夜夜夜 |