Java内存模型深度解析:总结--转
原文地址:http://www.codeceo.com/article/java-memory-7.html
處理器內(nèi)存模型
順序一致性內(nèi)存模型是一個理論參考模型,JMM和處理器內(nèi)存模型在設(shè)計(jì)時通常會把順序一致性內(nèi)存模型作為參照。JMM和處理器內(nèi)存模型在設(shè)計(jì)時會對順序一致性模型做一些放松,因?yàn)槿绻耆凑枕樞蛞恢滦阅P蛠韺?shí)現(xiàn)處理器和JMM,那么很多的處理器和編譯器優(yōu)化都要被禁止,這對執(zhí)行性能將會有很大的影響。
根據(jù)對不同類型讀/寫操作組合的執(zhí)行順序的放松,可以把常見處理器的內(nèi)存模型劃分為下面幾種類型:
注意,這里處理器對讀/寫操作的放松,是以兩個操作之間不存在數(shù)據(jù)依賴性為前提的(因?yàn)樘幚砥饕袷豠s-if-serial語義,處理器不會對存在數(shù)據(jù)依賴性的兩個內(nèi)存操作做重排序)。
下面的表格展示了常見處理器內(nèi)存模型的細(xì)節(jié)特征:
| 內(nèi)存模型名稱 | 對應(yīng)的處理器 ? | Store-Load 重排序 | Store-Store重排序 | Load-Load 和Load-Store重排序 | 可以更早讀取到其它處理器的寫 | 可以更早讀取到當(dāng)前處理器的寫 |
| TSO | sparc-TSOX64 | Y | ? | ? | ? | Y |
| PSO | sparc-PSO | Y | Y | ? | ? | Y |
| RMO | ia64 | Y | Y | Y | ? | Y |
| PowerPC | PowerPC | Y | Y | Y | Y | Y |
在這個表格中,我們可以看到所有處理器內(nèi)存模型都允許寫-讀重排序,原因在第一章以說明過:它們都使用了寫緩存區(qū),寫緩存區(qū)可能導(dǎo)致寫-讀操作重排序。同時,我們可以看到這些處理器內(nèi)存模型都允許更早讀到當(dāng)前處理器的寫,原因同樣是因?yàn)閷懢彺鎱^(qū):由于寫緩存區(qū)僅對當(dāng)前處理器可見,這個特性導(dǎo)致當(dāng)前處理器可以比其他處理器先看到臨時保存在自己的寫緩存區(qū)中的寫。
上面表格中的各種處理器內(nèi)存模型,從上到下,模型由強(qiáng)變?nèi)酢T绞亲非笮阅艿奶幚砥?#xff0c;內(nèi)存模型設(shè)計(jì)的會越弱。因?yàn)檫@些處理器希望內(nèi)存模型對它們的束縛越少越好,這樣它們就可以做盡可能多的優(yōu)化來提高性能。
由于常見的處理器內(nèi)存模型比JMM要弱,java編譯器在生成字節(jié)碼時,會在執(zhí)行指令序列的適當(dāng)位置插入內(nèi)存屏障來限制處理器的重排序。同時,由于各種處理器內(nèi)存模型的強(qiáng)弱并不相同,為了在不同的處理器平臺向程序員展示一個一致的內(nèi)存模型,JMM在不同的處理器中需要插入的內(nèi)存屏障的數(shù)量和種類也不相同。下圖展示了JMM在不同處理器內(nèi)存模型中需要插入的內(nèi)存屏障的示意圖:
如上圖所示,JMM屏蔽了不同處理器內(nèi)存模型的差異,它在不同的處理器平臺之上為java程序員呈現(xiàn)了一個一致的內(nèi)存模型。
JMM,處理器內(nèi)存模型與順序一致性內(nèi)存模型之間的關(guān)系
JMM是一個語言級的內(nèi)存模型,處理器內(nèi)存模型是硬件級的內(nèi)存模型,順序一致性內(nèi)存模型是一個理論參考模型。下面是語言內(nèi)存模型,處理器內(nèi)存模型和順序一致性內(nèi)存模型的強(qiáng)弱對比示意圖:
從上圖我們可以看出:常見的4種處理器內(nèi)存模型比常用的3中語言內(nèi)存模型要弱,處理器內(nèi)存模型和語言內(nèi)存模型都比順序一致性內(nèi)存模型要弱。同處理器內(nèi)存模型一樣,越是追求執(zhí)行性能的語言,內(nèi)存模型設(shè)計(jì)的會越弱。
JMM的設(shè)計(jì)
從JMM設(shè)計(jì)者的角度來說,在設(shè)計(jì)JMM時,需要考慮兩個關(guān)鍵因素:
- 程序員對內(nèi)存模型的使用。程序員希望內(nèi)存模型易于理解,易于編程。程序員希望基于一個強(qiáng)內(nèi)存模型來編寫代碼。
- 編譯器和處理器對內(nèi)存模型的實(shí)現(xiàn)。編譯器和處理器希望內(nèi)存模型對它們的束縛越少越好,這樣它們就可以做盡可能多的優(yōu)化來提高性能。編譯器和處理器希望實(shí)現(xiàn)一個弱內(nèi)存模型。
由于這兩個因素互相矛盾,所以JSR-133專家組在設(shè)計(jì)JMM時的核心目標(biāo)就是找到一個好的平衡點(diǎn):一方面要為程序員提供足夠強(qiáng)的內(nèi)存可見性保證;另一方面,對編譯器和處理器的限制要盡可能的放松。下面讓我們看看JSR-133是如何實(shí)現(xiàn)這一目標(biāo)的。
為了具體說明,請看前面提到過的計(jì)算圓面積的示例代碼:
double pi = 3.14; //A double r = 1.0; //B double area = pi * r * r; //C上面計(jì)算圓的面積的示例代碼存在三個happens- before關(guān)系:
由于A happens- before B,happens- before的定義會要求:A操作執(zhí)行的結(jié)果要對B可見,且A操作的執(zhí)行順序排在B操作之前。 但是從程序語義的角度來說,對A和B做重排序即不會改變程序的執(zhí)行結(jié)果,也還能提高程序的執(zhí)行性能(允許這種重排序減少了對編譯器和處理器優(yōu)化的束縛)。也就是說,上面這3個happens- before關(guān)系中,雖然2和3是必需要的,但1是不必要的。因此,JMM把happens- before要求禁止的重排序分為了下面兩類:
- 會改變程序執(zhí)行結(jié)果的重排序。
- 不會改變程序執(zhí)行結(jié)果的重排序。
JMM對這兩種不同性質(zhì)的重排序,采取了不同的策略:
- 對于會改變程序執(zhí)行結(jié)果的重排序,JMM要求編譯器和處理器必須禁止這種重排序。
- 對于不會改變程序執(zhí)行結(jié)果的重排序,JMM對編譯器和處理器不作要求(JMM允許這種重排序)。
下面是JMM的設(shè)計(jì)示意圖:
從上圖可以看出兩點(diǎn):
- JMM向程序員提供的happens- before規(guī)則能滿足程序員的需求。JMM的happens- before規(guī)則不但簡單易懂,而且也向程序員提供了足夠強(qiáng)的內(nèi)存可見性保證(有些內(nèi)存可見性保證其實(shí)并不一定真實(shí)存在,比如上面的A happens- before B)。
- JMM對編譯器和處理器的束縛已經(jīng)盡可能的少。從上面的分析我們可以看出,JMM其實(shí)是在遵循一個基本原則:只要不改變程序的執(zhí)行結(jié)果(指的是單線程程序和正確同步的多線程程序),編譯器和處理器怎么優(yōu)化都行。比如,如果編譯器經(jīng)過細(xì)致的分析后,認(rèn)定一個鎖只會被單個線程訪問,那么這個鎖可以被消除。再比如,如果編譯器經(jīng)過細(xì)致的分析后,認(rèn)定一個Volatile變量僅僅只會被單個線程訪問,那么編譯器可以把這個volatile變量當(dāng)作一個普通變量來對待。這些優(yōu)化既不會改變程序的執(zhí)行結(jié)果,又能提高程序的執(zhí)行效率。
JMM的內(nèi)存可見性保證
Java程序的內(nèi)存可見性保證按程序類型可以分為下列三類:
下圖展示了這三類程序在JMM中與在順序一致性內(nèi)存模型中的執(zhí)行結(jié)果的異同:
只要多線程程序是正確同步的,JMM保證該程序在任意的處理器平臺上的執(zhí)行結(jié)果,與該程序在順序一致性內(nèi)存模型中的執(zhí)行結(jié)果一致。
JSR-133對舊內(nèi)存模型的修補(bǔ)
JSR-133對JDK5之前的舊內(nèi)存模型的修補(bǔ)主要有兩個:
- 增強(qiáng)volatile的內(nèi)存語義。舊內(nèi)存模型允許volatile變量與普通變量重排序。JSR-133嚴(yán)格限制volatile變量與普通變量的重排序,使volatile的寫-讀和鎖的釋放-獲取具有相同的內(nèi)存語義。
- 增強(qiáng)final的內(nèi)存語義。在舊內(nèi)存模型中,多次讀取同一個final變量的值可能會不相同。為此,JSR-133為final增加了兩個重排序規(guī)則。現(xiàn)在,final具有了初始化安全性。
?
轉(zhuǎn)載于:https://www.cnblogs.com/davidwang456/p/6123145.html
《新程序員》:云原生和全面數(shù)字化實(shí)踐50位技術(shù)專家共同創(chuàng)作,文字、視頻、音頻交互閱讀總結(jié)
以上是生活随笔為你收集整理的Java内存模型深度解析:总结--转的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Java内存模型深度解析:final--
- 下一篇: 用枚举enum替代int常量