java記憶體模型-總結

来源:http://www.cnblogs.com/houziwty/archive/2016/08/10/5742493.html
-Advertisement-
Play Games

處理器記憶體模型 順序一致性記憶體模型是一個理論參考模型,JMM 和處理器記憶體模型在設計時通常會把順序一致性記憶體模型作為參照。JMM 和處理器記憶體模型在設計時會對順序一致性模型做一些放鬆,因為如果完全按照順序一致性模型來實現處理器和 JMM,那麼很多的處理器和編譯器優化都要被禁止,這對執行性能將會有很大 ...


處理器記憶體模型

順序一致性記憶體模型是一個理論參考模型,JMM 和處理器記憶體模型在設計時通常會把順序一致性記憶體模型作為參照。JMM 和處理器記憶體模型在設計時會對順序一致性模型做一些放鬆,因為如果完全按照順序一致性模型來實現處理器和 JMM,那麼很多的處理器和編譯器優化都要被禁止,這對執行性能將會有很大的影響。

根據對不同類型讀/寫操作組合的執行順序的放鬆,可以把常見處理器的記憶體模型劃分為下麵幾種類型:

  1. 放鬆程式中寫-讀操作的順序,由此產生了 total store ordering 記憶體模型(簡稱為TSO)。
  2. 在前面1的基礎上,繼續放鬆程式中寫-寫操作的順序,由此產生了 partial store order 記憶體模型(簡稱為 PSO)。
  3. 在前面1和2的基礎上,繼續放鬆程式中讀-寫和讀-讀操作的順序,由此產生了 relaxed memory order 記憶體模型(簡稱為 RMO)和 PowerPC 記憶體模型。

註意,這裡處理器對讀/寫操作的放鬆,是以兩個操作之間不存在數據依賴性為前提的(因為處理器要遵守 as-if-serial 語義,處理器不會對存在數據依賴性的兩個記憶體操作做重排序)。

下麵的表格展示了常見處理器記憶體模型的細節特征:

記憶體模型名稱 對應的處理器 Store-Load 重排序 Store-Store重排序 Load-Load 和Load-Store重排序 可以更早讀取到其它處理器的寫 可以更早讀取到當前處理器的寫
TSO sparc-TSOX64 Y       Y
PSO sparc-PSO Y Y     Y
RMO ia64 Y Y Y   Y
PowerPC PowerPC Y Y Y Y Y

在這個表格中,我們可以看到所有處理器記憶體模型都允許寫-讀重排序,原因在第一章以說明過:它們都使用了寫緩存區,寫緩存區可能導致寫-讀操作重排序。同時,我們可以看到這些處理器記憶體模型都允許更早讀到當前處理器的寫,原因同樣是因為寫緩存區:由於寫緩存區僅對當前處理器可見,這個特性導致當前處理器可以比其他處理器先看到臨時保存在自己的寫緩存區中的寫。

上面表格中的各種處理器記憶體模型,從上到下,模型由強變弱。越是追求性能的處理器,記憶體模型設計的會越弱。因為這些處理器希望記憶體模型對它們的束縛越少越好,這樣它們就可以做儘可能多的優化來提高性能。

由於常見的處理器記憶體模型比 JMM 要弱,java 編譯器在生成位元組碼時,會在執行指令序列的適當位置插入記憶體屏障來限制處理器的重排序。同時,由於各種處理器記憶體模型的強弱並不相同,為了在不同的處理器平臺向程式員展示一個一致的記憶體模型,JMM 在不同的處理器中需要插入的記憶體屏障的數量和種類也不相同。下圖展示了 JMM 在不同處理器記憶體模型中需要插入的記憶體屏障的示意圖:

如上圖所示,JMM 屏蔽了不同處理器記憶體模型的差異,它在不同的處理器平臺之上為 java 程式員呈現了一個一致的記憶體模型。

JMM,處理器記憶體模型與順序一致性記憶體模型之間的關係

JMM 是一個語言級的記憶體模型,處理器記憶體模型是硬體級的記憶體模型,順序一致性記憶體模型是一個理論參考模型。下麵是語言記憶體模型,處理器記憶體模型和順序一致性記憶體模型的強弱對比示意圖:

從上圖我們可以看出:常見的4種處理器記憶體模型比常用的3中語言記憶體模型要弱,處理器記憶體模型和語言記憶體模型都比順序一致性記憶體模型要弱。同處理器記憶體模型一樣,越是追求執行性能的語言,記憶體模型設計的會越弱。

JMM 的設計

從 JMM 設計者的角度來說,在設計 JMM 時,需要考慮兩個關鍵因素:

  • 程式員對記憶體模型的使用。程式員希望記憶體模型易於理解,易於編程。程式員希望基於一個強記憶體模型來編寫代碼。
  • 編譯器和處理器對記憶體模型的實現。編譯器和處理器希望記憶體模型對它們的束縛越少越好,這樣它們就可以做儘可能多的優化來提高性能。編譯器和處理器希望實現一個弱記憶體模型。

由於這兩個因素互相矛盾,所以 JSR-133 專家組在設計 JMM 時的核心目標就是找到一個好的平衡點:一方面要為程式員提供足夠強的記憶體可見性保證;另一方面,對編譯器和處理器的限制要儘可能的放鬆。下麵讓我們看看 JSR-133 是如何實現這一目標的。

為了具體說明,請看前面提到過的計算圓面積的示例代碼:

double pi  = 3.14;    //A
double r   = 1.0;     //B
double area = pi * r * r; //C  

 

上面計算圓的面積的示例代碼存在三個 happens- before 關係:

  1. A happens- before B;
  2. B happens- before C;
  3. A happens- before C;

由於 A happens- before B,happens- before 的定義會要求:A 操作執行的結果要對B可見,且 A 操作的執行順序排在B操作之前。 但是從程式語義的角度來說,對A和B做重排序即不會改變程式的執行結果,也還能提高程式的執行性能(允許這種重排序減少了對編譯器和處理器優化的束縛)。也就是說,上面這3個 happens- before 關係中,雖然2和3是必需要的,但1是不必要的。因此,JMM 把 happens- before 要求禁止的重排序分為了下麵兩類:

  • 會改變程式執行結果的重排序。
  • 不會改變程式執行結果的重排序。

JMM 對這兩種不同性質的重排序,採取了不同的策略:

  • 對於會改變程式執行結果的重排序,JMM 要求編譯器和處理器必須禁止這種重排序。
  • 對於不會改變程式執行結果的重排序,JMM 對編譯器和處理器不作要求(JMM 允許這種重排序)。

下麵是JMM的設計示意圖:

從上圖可以看出兩點:

  • JMM 向程式員提供的 happens- before 規則能滿足程式員的需求。JMM 的 happens- before 規則不但簡單易懂,而且也向程式員提供了足夠強的記憶體可見性保證(有些記憶體可見性保證其實並不一定真實存在,比如上面的 A happens- before B)。
  • JMM 對編譯器和處理器的束縛已經儘可能的少。從上面的分析我們可以看出,JMM 其實是在遵循一個基本原則:只要不改變程式的執行結果(指的是單線程程式和正確同步的多線程程式),編譯器和處理器怎麼優化都行。比如,如果編譯器經過細緻的分析後,認定一個鎖只會被單個線程訪問,那麼這個鎖可以被消除。再比如,如果編譯器經過細緻的分析後,認定一個 volatile 變數僅僅只會被單個線程訪問,那麼編譯器可以把這個 volatile 變數當作一個普通變數來對待。這些優化既不會改變程式的執行結果,又能提高程式的執行效率。

JMM 的記憶體可見性保證

Java 程式的記憶體可見性保證按程式類型可以分為下列三類:

  1. 單線程程式。單線程程式不會出現記憶體可見性問題。編譯器,runtime 和處理器會共同確保單線程程式的執行結果與該程式在順序一致性模型中的執行結果相同。
  2. 正確同步的多線程程式。正確同步的多線程程式的執行將具有順序一致性(程式的執行結果與該程式在順序一致性記憶體模型中的執行結果相同)。這是 JMM 關註的重點,JMM 通過限制編譯器和處理器的重排序來為程式員提供記憶體可見性保證。
  3. 未同步/未正確同步的多線程程式。JMM 為它們提供了最小安全性保障:線程執行時讀取到的值,要麼是之前某個線程寫入的值,要麼是預設值(0,null,false)。

下圖展示了這三類程式在 JMM 中與在順序一致性記憶體模型中的執行結果的異同:

只要多線程程式是正確同步的,JMM 保證該程式在任意的處理器平臺上的執行結果,與該程式在順序一致性記憶體模型中的執行結果一致。

JSR-133 對舊記憶體模型的修補

JSR-133 對 JDK5 之前的舊記憶體模型的修補主要有兩個:

  • 增強 volatile 的記憶體語義。舊記憶體模型允許 volatile 變數與普通變數重排序。JSR-133 嚴格限制 volatile 變數與普通變數的重排序,使 volatile 的寫-讀和鎖的釋放-獲取具有相同的記憶體語義。
  • 增強 final 的記憶體語義。在舊記憶體模型中,多次讀取同一個 final 變數的值可能會不相同。為此,JSR-133 為 final 增加了兩個重排序規則。現在,final 具有了初始化安全性。

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

-Advertisement-
Play Games
更多相關文章
  • 最近做一個項目要獲得ScrollBar的位置,因為.net找不到此類功能,只好用MFC中的函數了,GetScrollPos只返回listview頂部的位置,此時我找到了GetScrollInfo,覺得此函甚好。不成想從網上找到示例代碼後,函數執行成功了,但是返回了false,查下msdn,說是沒取到 ...
  • 一、前言 自己挖的坑還是得自己來填,當年學數據結構(C++版本)天天打醬油,課程結業的時候還以為->是一個字元,自己還納悶這東西是怎麼鍵入的,直到做結業設計的時候看團支書的代碼才突然醒悟,特此感謝下團支書MM,我想如果老師知道了應該不會打我...,後來嘗試看過兩次數據結構,都沒堅持看完。現找了一本C ...
  • 1. Integer 型變數 a 轉換成 String 時, 如果 a 是 null ,用 Integer.toString(a) 或者 a.toString() 都會報空指針異常,需要 放到 try catch 中捕獲異常。 如上代碼,如果 根據手機號 沒有查到 Userid ,則 Userid ...
  • some characters cannot be mapped using "Cp1251" character encoding. 解決辦法:方案一: eclipse->Window->Preferences->General->Content Types->Text->Java Propert ...
  • 一個優秀的軟體不會隨意的創建很銷毀線程,因為創建和銷毀線程需要耗費大量的CPU時間以及需要和記憶體做出大量的交互。因此JDK5提出了使用線程池,讓程式員把更多的精力放在業務邏輯上面,弱化對線程的開閉管理。 JDK提供了四種不同的線程池給程式員使用 首先使用線程池,需要用到ExecutorService ...
  • 我們編程的過程中大部分使用了很出色的ORM框架,例如:MyBatis,Hibernate,SpringJDBC,但是這些都離不開數據驅動JDBC的支持。雖然使用起來很方便,但是碰到一些問題確實很棘手,就比如困擾我一宿沒睡好覺的問題,jdbc生成執行數據,具體的我們看一下。 通常我們用MyBatis框 ...
  • 列表格式:name = []name = [name1, name2, name3, name4, name5] #針對列表的操作 #增加 add #刪除 delete #查詢 select #更改 update #列表copy分為深copy和淺copy 深copy 會把列表裡的子列表 copy過去 ...
  • Class Abstraction -- Object Interfaces 抽象類 對象介面 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...