欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

雙重檢查鎖定模式Java中的陷阱案例

 更新時間:2021年10月22日 10:21:13   作者:onlythinking  
這篇文章主要介紹了雙重檢查鎖定模式Java中的陷阱,雙重檢查鎖定(也叫做雙重檢查鎖定優(yōu)化)是一種軟件設(shè)計模式,它的作用是減少延遲初始化在多線程環(huán)境下獲取鎖的次數(shù),尤其是單例模式下比較突出,想具體了解的小伙伴可以參考下面文章內(nèi)容,附呦詳細的舉例說明

1、簡介

雙重檢查鎖定(也叫做雙重檢查鎖定優(yōu)化)是一種軟件設(shè)計模式。

它的作用是減少延遲初始化在多線程環(huán)境下獲取鎖的次數(shù),尤其是單例模式下比較突出。

  • 軟件設(shè)計模式:解決常用問題的通用解決方案。編程中針對一些常見業(yè)務(wù)固有的模版。
  • 延遲初始化:在編程中,將對象的創(chuàng)建,值計算或其他昂貴過程延遲到第一次使用時進行。
  • 單例模式:在一定范圍內(nèi),只生成一個實例對象。

2、Java中的雙重檢查鎖定

單例模式我們需保證實例只初始化一次。

下面例子在單線程環(huán)境奏效,多線程環(huán)境下會有線程安全問題(instance被初始化多次)。

private static Singleton instance;
public static Singleton getInstance() {
    if (null == instance) {
        instance = new Singleton();
    }
    return instance;
}

下面例子主要是性能問題。首先加鎖操作開銷很大,因為線程安全發(fā)生在對象初始化,而這里做了做了全局控制,造成浪費。

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è)有三個線程 T1 T2 T3 ,依次訪問 getInstance 方法。

  • T1 第一次檢查為Null 進入同步塊,T1持有鎖,第二次檢查為Null 執(zhí)行對象創(chuàng)建。
  • T2 第一次檢查為Null 進入同步塊,T2等待T1釋放鎖,鎖釋放后,T2進入執(zhí)行第二次檢查不為Null,返回實例對象。
  • T3 第一次檢查不為Null,直接返回對象。

上面一切似乎很完美,但是這里面存在陷阱。根據(jù)Java內(nèi)存模型我們知道,編譯器優(yōu)化處理會進行重排序。

instance = new Singleton() 大體分兩個步驟;

  • 1 創(chuàng)建初始化對象;
  • 2 引用賦值。

而 1 2 步驟可能顛倒,會造成對象屬性在初始化前調(diào)用的錯誤。

private static Singleton instance;
...
instance = new Singleton();
...
  
public class Singleton {
    private int age;
    public Singleton() {
        this.age = 80;
    }
}


這種細微的錯誤不容易出現(xiàn),但是它的確存在。大家可以參考下面這份報告,里面詳細記錄這個問題。

http://www.cs.umd.edu/~pugh/java/memoryModel/DoubleCheckedLocking.html

3、列舉方案

報告里面也列舉了幾種解決方案

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;
}

下面是不同方案下的性能比較報告

http://www.cs.umd.edu/~pugh/java/memoryModel/DCL-performance.html

4、總結(jié)

本章節(jié)主要記錄了雙重檢查鎖定模式使用中應(yīng)該注意的細微事項。

到此這篇關(guān)于雙重檢查鎖定模式Java中的陷阱案例的文章就介紹到這了,更多相關(guān)雙重檢查鎖定模式Java中的陷阱內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java.io.IOException:你的主機中的軟件中止了一個已建立的連接踩坑實戰(zhàn)

    java.io.IOException:你的主機中的軟件中止了一個已建立的連接踩坑實戰(zhàn)

    最近在工作中遇到了個問題,分享給同樣遇到問題的同學(xué),這篇文章主要給大家介紹了關(guān)于java.io.IOException:你的主機中的軟件中止了一個已建立的連接的踩坑實戰(zhàn)記錄,需要的朋友可以參考下
    2023-03-03
  • 攔截器獲取request的值之后,Controller拿不到值的解決

    攔截器獲取request的值之后,Controller拿不到值的解決

    這篇文章主要介紹了攔截器獲取request的值之后,Controller拿不到值的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • springboot下mybatis-plus開啟打印sql日志的配置指南

    springboot下mybatis-plus開啟打印sql日志的配置指南

    這篇文章主要給大家介紹了關(guān)于springboot下mybatis-plus開啟打印sql日志的配置指南的相關(guān)資料,還介紹了關(guān)閉打印的方法,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-03-03
  • 解讀Java中打印輸出對象內(nèi)容為什么可以不寫.toString()

    解讀Java中打印輸出對象內(nèi)容為什么可以不寫.toString()

    這篇文章主要介紹了解讀Java中打印輸出對象內(nèi)容為什么可以不寫.toString()問題,具有很的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • 詳解用JWT對SpringCloud進行認證和鑒權(quán)

    詳解用JWT對SpringCloud進行認證和鑒權(quán)

    這篇文章主要介紹了詳解用JWT對SpringCloud進行認證和鑒權(quán),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • java模擬ATM功能(控制臺連接Mysql數(shù)據(jù)庫)

    java模擬ATM功能(控制臺連接Mysql數(shù)據(jù)庫)

    這篇文章主要介紹了java模擬ATM功能,控制臺連接Mysql數(shù)據(jù)庫,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-05-05
  • 如何使用SpringSecurity保護程序安全

    如何使用SpringSecurity保護程序安全

    這篇文章主要介紹了如何使用SpringSecurity保護程序安全,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-09-09
  • Springboot主程序類注解配置過程圖解

    Springboot主程序類注解配置過程圖解

    這篇文章主要介紹了Springboot主程序類注解配置過程圖解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-10-10
  • 解析java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別

    解析java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別

    這篇文章主要介紹了java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-05-05
  • 一篇文章帶你入門Java?UML的類圖

    一篇文章帶你入門Java?UML的類圖

    這篇文章主要為大家詳細介紹了Java?UML的類圖,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-02-02

最新評論