日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

双重检查锁实现单例模式的线程安全问题

發布時間:2023/12/20 编程问答 37 豆豆
生活随笔 收集整理的這篇文章主要介紹了 双重检查锁实现单例模式的线程安全问题 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、結論?

雙重校驗鎖的單例模式代碼如下:

public class Singleton {
  private static Singleton singleton;

  private Singleton() {}

  public static Singleton getSingleton() {
    if (singleton == null) { // 1
      synchronized (Singleton.class) { // 2
        if (singleton == null) { // 3
          singleton = new Singleton(); // 4
        }
      }
    }
    return singleton;
  }
}

假設有兩個線程AB同時訪問上面這段代碼,它并不能保證線程安全。

二、問題說明

  1、指令重排序的簡單說明

  重排序是指編譯器和處理器為了優化程序性能而對指令序列進行重新排序的一種手段。

  (1)編譯器指令重排序

    編譯器在不改變程序 執行結果的前提下,可以對程序的執行順序進行優化重新排序

  ? ?(2)?處理器指令重排序

    參考:https://blog.csdn.net/javazejian/article/details/72772461 (處理器指令重排)

  2、 對象創建過程

    分為三步,如下圖:

? ? ? ? ? ? ?

我們認為程序應該是按照1、2、3的步驟走下去,但實際上可能不是這樣的,這里編譯器和處理器可能會對2、3步的執行順序進行重排序,即先將對象的引用指向內存空間,實際上A線程返回的是沒有初始化的對象,然后B線程訪問上面這段代碼,判斷if (singleton == null) { // 1 就為false,它會認為Singleton類已經實例化,問題就出在這里。

  重排序后A 、B線程執行時序圖如下:

? ? ?

?三、解決方案

  1、不允許對象創建過程中2、3步發生指令重排序 (基于volatile的解決方案)

   即將Singleton聲明時加上volatile,volatile關鍵字可以保證內存可見性和禁止指令重排序,關于volatile參見https://blog.csdn.net/javazejian/article/details/72772461 (volatile內存語義)

    修改后的代碼:

    public class Singleton {
      private volatile static Singleton singleton;

      private Singleton() {}

      public static Singleton getSingleton() {
        if (singleton == null) { // 1
          synchronized (Singleton.class) { // 2
            if (singleton == null) { // 3
              singleton = new Singleton(); // 4
            }
          }
        }
        return singleton;
      }
    }

    Singleton屬性被加上volatile后,4中對象創建過程的2、3兩步在多線程環境下就被禁止重排序,這樣就能保證線程安全。

  2、允許對象創建過程中2、3重排序,但不允許其他線程看到這個重排序 (基于類初始化的解決方案)

    JVM在類的初始化階段,會執行類的初始化。在執行類的初始化期間,JVM會獲取一個鎖,這個鎖可以同步多個線程對同一個類的初始化。基于這個特性修改代碼如下:

    public class Singleton {
      private static class SingletonHolder{
        public static Singleton singleton = new Singleton();
      }
      public static Singleton getSingleton(){
        return SingletonHolder.singleton;
      }
    }

  ? ? ? 多線程訪問上面這段程序的時序圖如下:

   

?

?參考資料:

  1、https://blog.csdn.net/javazejian/article/details/72772461

  2、《Java并發編程的藝術》第三章 Java內存模型

    

說明:菜鳥一枚,第一次發技術博客,如有錯誤或者寫的不好的地方歡迎大家指正。

轉載于:https://www.cnblogs.com/-Marksman/p/9219274.html

總結

以上是生活随笔為你收集整理的双重检查锁实现单例模式的线程安全问题的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。