雙重檢查鎖定模式Java中的陷阱案例
1、簡介
雙重檢查鎖定(也叫做雙重檢查鎖定優(yōu)化)是一種軟件設(shè)計(jì)模式。
它的作用是減少延遲初始化在多線程環(huán)境下獲取鎖的次數(shù),尤其是單例模式下比較突出。
- 軟件設(shè)計(jì)模式:解決常用問題的通用解決方案。編程中針對(duì)一些常見業(yè)務(wù)固有的模版。
- 延遲初始化:在編程中,將對(duì)象的創(chuàng)建,值計(jì)算或其他昂貴過程延遲到第一次使用時(shí)進(jìn)行。
- 單例模式:在一定范圍內(nèi),只生成一個(gè)實(shí)例對(duì)象。
2、Java中的雙重檢查鎖定
單例模式我們需保證實(shí)例只初始化一次。
下面例子在單線程環(huán)境奏效,多線程環(huán)境下會(huì)有線程安全問題(instance被初始化多次)。
private static Singleton instance;
public static Singleton getInstance() {
if (null == instance) {
instance = new Singleton();
}
return instance;
}
下面例子主要是性能問題。首先加鎖操作開銷很大,因?yàn)榫€程安全發(fā)生在對(duì)象初始化,而這里做了做了全局控制,造成浪費(fèi)。
public synchronized static Singleton getInstance() {
if (null == instance) {
instance = new Singleton();
}
return instance;
}
為了控制線程安全又能保證性能,雙重檢查鎖定模式出現(xiàn)。
public static Singleton getInstance() {
if (null == instance) {
synchronized (Singleton.class) {
if (null == instance) {
instance = new Singleton();
}
}
}
return instance;
}
邏輯如下:

我們分析一下執(zhí)行邏輯:
假設(shè)有三個(gè)線程 T1 T2 T3 ,依次訪問 getInstance 方法。
T1第一次檢查為Null進(jìn)入同步塊,T1持有鎖,第二次檢查為Null 執(zhí)行對(duì)象創(chuàng)建。T2第一次檢查為Null進(jìn)入同步塊,T2等待T1釋放鎖,鎖釋放后,T2進(jìn)入執(zhí)行第二次檢查不為Null,返回實(shí)例對(duì)象。T3第一次檢查不為Null,直接返回對(duì)象。

上面一切似乎很完美,但是這里面存在陷阱。根據(jù)Java內(nèi)存模型我們知道,編譯器優(yōu)化處理會(huì)進(jìn)行重排序。
instance = new Singleton() 大體分兩個(gè)步驟;
- 1 創(chuàng)建初始化對(duì)象;
- 2 引用賦值。
而 1 2 步驟可能顛倒,會(huì)造成對(duì)象屬性在初始化前調(diào)用的錯(cuò)誤。
private static Singleton instance;
...
instance = new Singleton();
...
public class Singleton {
private int age;
public Singleton() {
this.age = 80;
}
}
這種細(xì)微的錯(cuò)誤不容易出現(xiàn),但是它的確存在。大家可以參考下面這份報(bào)告,里面詳細(xì)記錄這個(gè)問題。

http://www.cs.umd.edu/~pugh/java/memoryModel/DoubleCheckedLocking.html
3、列舉方案
報(bào)告里面也列舉了幾種解決方案
3.1 利用 ThreadLocal
private static final ThreadLocal<Singleton> threadInstance = new ThreadLocal<>();
public static Singleton getInstance() {
if (null == threadInstance.get()) {
createInstance();
}
return instance;
}
private static void createInstance() {
synchronized (Singleton.class) {
if (instance == null)
instance = new Singleton();
}
threadInstance.set(instance);
}
3.2 利用volatile(解決重排序問題)
private volatile static Singleton instance;
public static Singleton getInstance() {
if (null == instance) {
synchronized (Singleton.class) {
if (null == instance) {
instance = new Singleton();
}
}
}
return instance;
}
下面是不同方案下的性能比較報(bào)告
http://www.cs.umd.edu/~pugh/java/memoryModel/DCL-performance.html
4、總結(jié)
本章節(jié)主要記錄了雙重檢查鎖定模式使用中應(yīng)該注意的細(xì)微事項(xiàng)。
到此這篇關(guān)于雙重檢查鎖定模式Java中的陷阱案例的文章就介紹到這了,更多相關(guān)雙重檢查鎖定模式Java中的陷阱內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
OpenFeign實(shí)現(xiàn)微服務(wù)間的文件下載方式
這篇文章主要介紹了OpenFeign實(shí)現(xiàn)微服務(wù)間的文件下載方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05
JAVA實(shí)現(xiàn)經(jīng)典游戲坦克大戰(zhàn)的示例代碼
小時(shí)候大家都玩過坦克大戰(zhàn)吧,熟悉的旋律和豐富的關(guān)卡陪伴了我們一整個(gè)寒暑假。本文將通過Java+Swing實(shí)現(xiàn)這一經(jīng)典游戲,感興趣的可以學(xué)習(xí)一下2022-01-01
SpringBoot集成Redisson實(shí)現(xiàn)消息隊(duì)列的示例代碼
本文介紹了如何在SpringBoot中通過集成Redisson來實(shí)現(xiàn)消息隊(duì)列的功能,包括RedisQueue、RedisQueueInit、RedisQueueListener、RedisQueueService等相關(guān)組件的實(shí)現(xiàn)和測試,感興趣的可以了解一下2024-10-10
Java實(shí)現(xiàn)文件上傳服務(wù)器和客戶端
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)文件上傳服務(wù)器和客戶端,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
Spring?Security?中多個(gè)身份驗(yàn)證的示例代碼
這篇文章主要介紹了Spring?Security?中多個(gè)身份驗(yàn)證的示例代碼,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09
Java中的 FilterInputStream簡介_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
FilterInputStream 的作用是用來“封裝其它的輸入流,并為它們提供額外的功能”。接下來通過本文給大家分享Java中的 FilterInputStream簡介,感興趣的朋友一起學(xué)習(xí)吧2017-05-05
SpringBoot配置加載,各配置文件優(yōu)先級(jí)對(duì)比方式
這篇文章主要介紹了SpringBoot配置加載,各配置文件優(yōu)先級(jí)對(duì)比方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08

