看完這篇,還不懂JAVA記憶體模型(JMM)算我輸

来源:https://www.cnblogs.com/alvinscript/archive/2022/12/06/16960418.html
-Advertisement-
Play Games

歡迎關註專欄【JAVA併發】 前言 開篇一個例子,我看看都有誰會?如果不會的,或者不知道原理的,還是老老實實看完這篇文章吧。 @Slf4j(topic = "c.VolatileTest") public class VolatileTest { static boolean run = true; ...


歡迎關註專欄【JAVA併發】

前言

開篇一個例子,我看看都有誰會?如果不會的,或者不知道原理的,還是老老實實看完這篇文章吧。

@Slf4j(topic = "c.VolatileTest")
public class VolatileTest {
    
    static boolean run = true;

    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(() -> {
            while (run) {
                // do other things
            }
            
            // ?????? 這行會列印嗎?
            log.info("done .....");
        });
        t.start();
        
        Thread.sleep(1000);

       // 設置run = false
        run = false;
    }
}

main函數中新開個線程根據標位run迴圈,主線程中sleep一秒,然後設置run=false,大家認為會列印"done ......."嗎?

答案就是不會列印,為什麼呢?

JAVA併發三大特性

我們先來解釋下上面問題的原因,如下圖所示,

現代的CPU架構基本有多級緩存機制,t線程會將run載入到高速緩存中,然後主線程修改了主記憶體的值為false,導致緩存不一致,但是t線程依然是從工作記憶體中的高速緩存讀取run的值,最終無法跳出迴圈。

可見性

正如上面的例子,由於不做任何處理,一個線程能否立刻看到另外一個線程修改的共用變數值,我們稱為"可見性"。

如果在併發程式中,不做任何處理,那麼就會帶來可見性問題,具體如何處理,見後文。

有序性

有序性是指程式按照代碼的先後順序執行。但是編譯器或者處理器出於性能原因,改變程式語句的先後順序,比如代碼順序"a=1; b=2;",但是指令重排序後,有可能會變成"b=2;a=1", 那麼這樣在併發情況下,會有問題嗎?

在單線程情況下,指令重排序不會有任何影響。但是在併發情況下,可能會導致一些意想不到的bug。比如下麵的例子:

public class Singleton {
  static Singleton instance;
    
  static Singleton getInstance(){
    if (instance == null) {
      synchronized(Singleton.class) {
        if (instance == null)
          instance = new Singleton();
        }
    }
    return instance;
  }
}

假設有兩個線程 A、B 同時調用 getInstance() 方法,正常情況下,他們都可以拿到instance實例。

但往往bug就在一些極端的異常情況,比如new Singleton() 這個操作,實際會有下麵3個步驟:

  1. 分配一塊記憶體 M;

  2. 在記憶體 M 上初始化 Singleton 對象;

  3. 然後 M 的地址賦值給 instance 變數。

現在發生指令重排序,順序變為下麵的方式:

  1. 分配一塊記憶體 M;

  2. 將 M 的地址賦值給 instance 變數;

  3. 最後在記憶體 M 上初始化 Singleton 對象。

優化後會導致什麼問題呢?我們假設線程 A 先執行 getInstance() 方法,當執行完指令 2 時恰好發生了線程切換,切換到了線程 B 上;如果此時線程 B 也執行 getInstance() 方法,那麼線程 B 在執行第一個判斷時會發現 instance != null ,所以直接返回 instance,而此時的 instance 是沒有初始化過的,如果我們這個時候訪問 instance 的成員變數就可能觸發空指針異常。

這就是併發情況下,有序性帶來的一個問題,這種情況又該如何處理呢?

當然,指令重排序並不會瞎排序,處理器在進行重排序時,必須要考慮指令之間的數據依賴性。

原子性

如上圖所示,在多線程的情況下,CPU資源會在不同的線程間切換。那麼這樣也會導致意向不到的問題。

比如你認為的一行代碼:count += 1,實際上涉及了多條CPU指令:

  • 指令 1:首先,需要把變數 count 從記憶體載入到 CPU 的寄存器;
  • 指令 2:之後,在寄存器中執行 +1 操作;
  • 指令 3:最後,將結果寫入記憶體(緩存機制導致可能寫入的是 CPU 緩存而不是記憶體)。

操作系統做任務切換,可以發生在任何一條CPU 指令執行完。假設 count=0,如果線程 A 在指令 1 執行完後做線程切換,線程 A 和線程 B 按照下圖的序列執行,那麼我們會發現兩個線程都執行了 count+=1 的操作,但是得到的結果不是我們期望的 2,而是 1。

我們潛意識認為的這個count+=1操作是一個不可分割的整體,就像一個原子一樣,我們把一個或者多個操作在 CPU 執行的過程中不被中斷的特性稱為原子性。但實際情況就是不做任何處理的話,在併發情況下CPU進行切換,導致出現原子性的問題,我們一般通過加鎖解決,這個不是本文的重點。

Java記憶體模型真面目

前面講解併發的三大特性,其中原子性問題可以通過加鎖的方式解決,那麼可見性和有序性有什麼解決的方案呢?其實也很容易想到,可見性是因為緩存導致,有序性是因為編譯優化指令重排序導致,那麼是不是可以讓程式員按需禁用緩存以及編譯優化, 因為只有程式員知道什麼情況下會出現問題 順著這個思路,就提出了JAVA記憶體模型(JMM)規範

Java 記憶體模型是 Java Memory Model(JMM),本身是一種抽象的概念,實際上並不存在,描述的是一組規則規範,通過這組規範定義了程式中各個變數(包括實例欄位,靜態欄位和構成數組對象的元素)的訪問方式。

預設情況下,JMM中的記憶體機制如下:

  • 系統存在一個主記憶體(Main Memory),Java 中所有變數都存儲在主存中,對於所有線程都是共用的
  • 每條線程都有自己的工作記憶體(Working Memory),工作記憶體中保存的是主存中某些變數的拷貝
  • 線程對所有變數的操作都是先對變數進行拷貝,然後在工作記憶體中進行,不能直接操作主記憶體中的變數
  • 線程之間無法相互直接訪問,線程間的通信(傳遞)必須通過主記憶體來完成

同時,JMM規範了 JVM 如何提供按需禁用緩存和編譯優化的方法,主要是通過volatilesynchronizedfinal 三個關鍵字,那具體的規則是什麼樣的呢?

JMM 中的主記憶體、工作記憶體與 JVM 中的 Java 堆、棧、方法區等並不是同一個層次的記憶體劃分,這兩者基本上是沒有關係的。

Happens-Before規則

JMM本質上包含了一些規則,那這個規則就是大家有所耳聞的Happens-Before規則,大家都理解了些規則嗎?

Happens-Before規則,可以簡單理解為如果想要A線程發生在B線程前面,也就是B線程能夠看到A線程,需要遵循6個原則。如果不符合 happens-before 規則,JMM 並不能保證一個線程的可見性和有序性。

1.程式的順序性規則

在一個線程中,邏輯上書寫在前面的操作先行發生於書寫在後面的操作。

這個規則很好理解,同一個線程中他們是用的同一個工作緩存,是可見的,並且多個操作之間有先後依賴關係,則不允許對這些操作進行重排序。

2. volatile 變數規則

指對一個 volatile 變數的寫操作, Happens-Before 於後續對這個 volatile 變數的讀操作。

怎麼理解呢?比如線程A對volatile變數進行寫操作,那麼線程B讀取這個volatile變數是可見的,就是說能夠讀取到最新的值。

3.傳遞性

這條規則是指如果 A Happens-Before B,且 B Happens-Before C,那麼 A Happens-Before C

這個規則也比較容易理解,不展開討論了。

  1. 鎖的規則

這條規則是指對一個鎖的解鎖 Happens-Before於後續對這個鎖的加鎖,這裡的鎖要是同一把鎖, 而且用synchronized或者ReentrantLock都可以。

如下代碼的例子:

synchronized (this) { // 此處自動加鎖
  // x 是共用變數, 初始值 =10
  if (this.x < 12) {
    this.x = 12; 
  }  
} // 此處自動解鎖
  • 假設 x 的初始值是 8,線程 A 執行完代碼塊後 x 的值會變成 12(執行完自動釋放鎖)
  • 線程 B 進入代碼塊時,能夠看到線程 A 對 x 的寫操作,也就是線程 B 能夠看到 x==12

5.線程 start() 規則

主線程 A 啟動子線程 B 後,子線程 B 能夠看到主線程在啟動子線程 B 前的操作。

這個規則也很容易理解,線程 A 調用線程 B 的 start() 方法(即線上程 A 中啟動線程 B),那麼該 start() 操作 Happens-Before 於線程 B 中的任意操作。

6.線程 join() 規則

線程 A 中,調用線程 B 的 join() 併成功返回,那麼線程 B 中的任意操作 Happens-Before 於該 join() 操作的返回。

使用JMM規則

我們現在已經基本講清楚了JAVA記憶體模型規範,以及裡面關鍵的Happens-Before規則,那有啥用呢?回到前言的問題中,我們是不是可以使用目前學到的關於JMM的知識去解決這個問題。

方案一: 使用volatile

根據JMM的第2條規則,主線程寫了volatile修飾的run變數,後面的t線程讀取的時候就可以看到了。

方案二:使用鎖

利用synchronized鎖的規則,主線程釋放鎖,那麼後續t線程加鎖就可以看到之前的內容了。

小結:

volatile 關鍵字

  • 保證可見性
  • 不保證原子性
  • 保證有序性(禁止指令重排)

volatile 修飾的變數進行讀操作與普通變數幾乎沒什麼差別,但是寫操作相對慢一些,因為需要在本地代碼中插入很多記憶體屏障來保證指令不會發生亂序執行,但是開銷比鎖要小。volatile的性能遠比加鎖要好。

synchronized 關鍵字

  • 保證可見性
  • 不保證原子性
  • 保證有序性

加了鎖之後,只能有一個線程獲得到了鎖,獲得不到鎖的線程就要阻塞,所以同一時間只有一個線程執行,相當於單線程,由於數據依賴性的存在,單線程的指令重排是沒有問題的。

線程加鎖前,將清空工作記憶體中共用變數的值,使用共用變數時需要從主記憶體中重新讀取最新的值;線程解鎖前,必須把共用變數的最新值刷新到主記憶體中。

總結

本文講解了JAVA併發的3大特性,可見性、有序性和原子性。從而引出了JAVA記憶體模型規範,這主要是為瞭解決併發情況下帶來的可見性和有序性問題,主要就是定義了一些規則,需要我們程式員懂得這些規則,然後根據實際場景去使用,就是使用volatilesynchronizedfinal關鍵字,主要final關鍵字也會讓其他線程可見,並且保證有序性。那麼具體他們底層的實現是什麼,是如何保證可見和有序的,我們後面詳細講解。

如果本文對你有幫助的話,請留下一個贊吧
更多技術幹活和學習資料盡在個人公眾號——JAVA旭陽

本文來自博客園,作者:JAVA旭陽,轉載請註明原文鏈接:https://www.cnblogs.com/alvinscript/p/16960418.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 因公司業務需要,需要開發水印相機功能,而項目代碼用的uniapp框架,App端只能簡單調用系統的相機,無法自定義界面,在此基礎上,只能開發自定義插件來完成功能(自定義原生插件,即是用原生代碼來編寫組件實現功能,然後供uniapp項目調用) ...
  • Type 描述:全稱叫做 '類型別名',為類型字面量提供名稱。比 Interface 支持更豐富的類型系統特性。 Type 與 Interface 區別 Interface 只能描述對象的形狀,Type 不止 Interface 能多次聲明進行擴展,Type 不行 在性能方面,Type 介面檢查能夠 ...
  • 在過往,我們想要實現一個圖片的漸隱消失。最常見的莫過於整體透明度的變化,像是這樣: <div class="img"></div> div { width: 300px; height: 300px; background: url(image.jpg); transition: .4s; } .i ...
  • 我們是袋鼠雲數棧 UED 團隊,致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。 本文作者:正則 作為一名前端開發人員,平時開發中使用最多的就是 Chrome devtools,但可能很多同學像我一樣平時用的最多也就 Console、Elements ...
  • 軟體設計模式(Design pattern),又稱設計模式,是一套被反覆使用、多數人知曉的、經過分類編目的、代碼設計經驗的總結。使用設計模式是為了可重用代碼、讓代碼更容易被他人理解、保證代碼可靠性、程式的重用性。 ...
  • 摘要:華為雲Solution as Code推出基於Ploto構建自動駕駛平臺解決方案。 本文分享自華為雲社區《基於Ploto構建自動駕駛平臺》,作者:阿米托福 。 2022年6月15日,主題為“因聚而生 為你所能”的華為伙伴暨開發者大會 2022 正式開啟,在自動駕駛專場中,華為雲攜手合作伙伴聯合 ...
  • JSON&Ajax01 JSON 線上文檔 AJAX 線上文檔 1.JSON介紹 JSON指的是JavaScript對象表示法( JavaScript Object Notation),JSON的本質仍然是JavaScript對象 JSON是輕量級的文本數據交互格式,也是前後端進行數據通訊的一種格式 ...
  • pycharm社區版可用於商業項目 pycharm社區版可用於商業項目,來源於官方的回答:Can I use Community Editions of JetBrains IDEs for developing commercial proprietary software? – Licensin ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...