Java記憶體模型深度解析:總結
處理器記憶體模型
順序一致性記憶體模型是一個理論參考模型,JMM和處理器記憶體模型在設計時通常會把順序一致性記憶體模型當作參考。 JMM和處理器記憶體模型在設計時會對順序一致性模型做一些放鬆,因為如果完全按照順序一致性模型來實現處理器和JMM,那麼很多的處理器和編譯器優化都要被禁止,這對執行性能將會有很大的影響。
根據對不同類型讀/寫操作組合的執行順序的放鬆,可以把常見處理器的內存模型劃分為下面幾種類型:
放鬆程序中寫-讀操作的順序,由此產生了total store ordering記憶體模型(簡稱TSO)。
在前面1的基礎上,繼續放鬆程序中寫入-寫入操作的順序,由此產生了partial store order 記憶體模型(簡稱PSO)。
在前面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 Y
Y
Y
PowerPC
PowerPC
Y
Y
Y
Y
Y
在這個表格中,我們可以看到所有處理器記憶體模型都允許寫入-讀重排序,原因在第一章以說明過:它們都使用了寫入快取區,寫入快取區可能導致寫入-讀取操作重新排序。同時,我們可以看到這些處理器記憶體模型都允許更早讀到目前處理器的寫,原因同樣是因為寫快取區:由於寫入快取區僅對當前處理器可見,這個特性導致當前處理器可以比其他處理器先看到臨時保存在自己的寫入快取區中的寫入。
上面表格中的各種處理器記憶體模型,從上到下,模型由強變弱。越是追求效能的處理器,記憶體模型設計的會越弱。因為這些處理器希望記憶體模型對它們的束縛越少越好,這樣它們就可以做盡可能多的最佳化來提高效能。
由於常見的處理器記憶體模型比JMM弱,java編譯器在產生字節碼時,會在執行指令序列的適當位置插入記憶體屏障來限制處理器的重新排序。同時,由於各種處理器記憶體模型的強度並不相同,為了在不同的處理器平台向程式設計師展示一個一致的記憶體模型,JMM在不同的處理器中需要插入的記憶體屏障的數量和種類也不相同。下圖展示了JMM在不同處理器記憶體模型中需要插入的記憶體屏障的示意圖:
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關係:
A happens- before B;
B happens- before C;
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程式的記憶體可見性保證依程式類型可分為下列三類:
單執行緒程式。單線程程式不會出現記憶體可見性問題。編譯器,runtime和處理器會共同確保單執行緒程式的執行結果與該程式在順序一致性模型中的執行結果相同。
正確同步的多執行緒程式。正確同步的多執行緒程式的執行將具有順序一致性(程式的執行結果與該程式在順序一致性記憶體模型中的執行結果相同)。這是JMM的重點,JMM透過限制編譯器和處理器的重排序來為程式設計師提供記憶體可見性保證。
未同步/未正確同步的多執行緒程式。 JMM為它們提供了最小安全性保障:執行緒執行時讀取到的值,要麼是先前某個執行緒寫入的值,要麼是預設值(0,null,false)。
下圖展示了這三類程式在JMM中與在順序一致性記憶體模型中的執行結果的異同:
只要多執行緒程式是正確同步的,JMM保證該程式在任意的處理器平台上的執行結果,與該程式在順序一致性記憶體模型中的執行結果一致。
JSR-133對舊內存模型的修補
JSR-133對JDK5之前的舊內存模型的修補主要有兩個:
增強volatile的內存語義。舊記憶體模型允許volatile變數與普通變數重排序。 JSR-133嚴格限制volatile變數與普通變數的重新排序,使volatile的寫-讀和鎖的釋放-獲取具有相同的記憶體語義。
增強final的記憶語語義。在舊記憶體模型中,多次讀取同一個final變數的值可能會不相同。為此,JSR-133為final增加了兩個重排序規則。現在,final具有了初始化安全性。
以上就是Java記憶體模型深度解析:總結的內容,更多相關內容請關注PHP中文網(www.php.cn)!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

Java 8引入了Stream API,提供了一種強大且表達力豐富的處理數據集合的方式。然而,使用Stream時,一個常見問題是:如何從forEach操作中中斷或返回? 傳統循環允許提前中斷或返回,但Stream的forEach方法並不直接支持這種方式。本文將解釋原因,並探討在Stream處理系統中實現提前終止的替代方法。 延伸閱讀: Java Stream API改進 理解Stream forEach forEach方法是一個終端操作,它對Stream中的每個元素執行一個操作。它的設計意圖是處

PHP是一種廣泛應用於服務器端的腳本語言,特別適合web開發。 1.PHP可以嵌入HTML,處理HTTP請求和響應,支持多種數據庫。 2.PHP用於生成動態網頁內容,處理表單數據,訪問數據庫等,具有強大的社區支持和開源資源。 3.PHP是解釋型語言,執行過程包括詞法分析、語法分析、編譯和執行。 4.PHP可以與MySQL結合用於用戶註冊系統等高級應用。 5.調試PHP時,可使用error_reporting()和var_dump()等函數。 6.優化PHP代碼可通過緩存機制、優化數據庫查詢和使用內置函數。 7

PHP和Python各有優勢,選擇應基於項目需求。 1.PHP適合web開發,語法簡單,執行效率高。 2.Python適用於數據科學和機器學習,語法簡潔,庫豐富。

PHP適合web開發,特別是在快速開發和處理動態內容方面表現出色,但不擅長數據科學和企業級應用。與Python相比,PHP在web開發中更具優勢,但在數據科學領域不如Python;與Java相比,PHP在企業級應用中表現較差,但在web開發中更靈活;與JavaScript相比,PHP在後端開發中更簡潔,但在前端開發中不如JavaScript。

PHP和Python各有優勢,適合不同場景。 1.PHP適用於web開發,提供內置web服務器和豐富函數庫。 2.Python適合數據科學和機器學習,語法簡潔且有強大標準庫。選擇時應根據項目需求決定。

膠囊是一種三維幾何圖形,由一個圓柱體和兩端各一個半球體組成。膠囊的體積可以通過將圓柱體的體積和兩端半球體的體積相加來計算。本教程將討論如何使用不同的方法在Java中計算給定膠囊的體積。 膠囊體積公式 膠囊體積的公式如下: 膠囊體積 = 圓柱體體積 兩個半球體體積 其中, r: 半球體的半徑。 h: 圓柱體的高度(不包括半球體)。 例子 1 輸入 半徑 = 5 單位 高度 = 10 單位 輸出 體積 = 1570.8 立方單位 解釋 使用公式計算體積: 體積 = π × r2 × h (4

PHP成為許多網站首選技術棧的原因包括其易用性、強大社區支持和廣泛應用。 1)易於學習和使用,適合初學者。 2)擁有龐大的開發者社區,資源豐富。 3)廣泛應用於WordPress、Drupal等平台。 4)與Web服務器緊密集成,簡化開發部署。

Java是熱門程式語言,適合初學者和經驗豐富的開發者學習。本教學從基礎概念出發,逐步深入解說進階主題。安裝Java開發工具包後,可透過建立簡單的「Hello,World!」程式來實踐程式設計。理解程式碼後,使用命令提示字元編譯並執行程序,控制台上將輸出「Hello,World!」。學習Java開啟了程式設計之旅,隨著掌握程度加深,可創建更複雜的應用程式。
