一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專(zhuān)注于服務(wù)器技術(shù)及軟件下載分享
分類(lèi)導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|JavaScript|易語(yǔ)言|

服務(wù)器之家 - 編程語(yǔ)言 - Java教程 - Java中鎖的實(shí)現(xiàn)和內(nèi)存語(yǔ)義淺析

Java中鎖的實(shí)現(xiàn)和內(nèi)存語(yǔ)義淺析

2021-06-15 09:46蝸牛大師 Java教程

這篇文章主要給大家介紹了關(guān)于Java中鎖的實(shí)現(xiàn)和內(nèi)存語(yǔ)義的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

1. 概述

鎖是java并發(fā)編程中最重要的同步機(jī)制。鎖除了讓臨界區(qū)互斥執(zhí)行外,還可以讓釋放鎖的線程獲取同一個(gè)鎖的線程發(fā)送消息。

鎖在實(shí)際使用時(shí)只是明白鎖限制了并發(fā)訪問(wèn), 但是鎖是如何實(shí)現(xiàn)并發(fā)訪問(wèn)的, 同學(xué)們可能不太清楚, 下面這篇文章就來(lái)揭開(kāi)鎖的神秘面紗.

2. 鎖的內(nèi)存語(yǔ)義

  • 當(dāng)線程獲取鎖時(shí), jmm會(huì)把線程對(duì)應(yīng)的本地內(nèi)存置為無(wú)效. 從而使得被監(jiān)視器保護(hù)的臨界區(qū)的變量必須從主內(nèi)存中讀取.
  • 當(dāng)線程釋放鎖時(shí), jmm會(huì)把該線程對(duì)應(yīng)的本地內(nèi)存中的共享變量刷新到主內(nèi)存中(并不是不釋放鎖就不刷新到主內(nèi)存, 只是釋放鎖時(shí)把未刷新到主內(nèi)存中的數(shù)據(jù)刷新到主內(nèi)存).

鎖的內(nèi)存語(yǔ)義與volatile的內(nèi)存語(yǔ)義

  • 鎖獲取與volatile讀有相同的內(nèi)存語(yǔ)義.
  • 鎖釋放與volatile寫(xiě)有相同的內(nèi)存語(yǔ)義.

內(nèi)存語(yǔ)義總結(jié)

  • 線程a釋放一個(gè)鎖, 實(shí)質(zhì)上是線程a向接下來(lái)將要獲取這個(gè)鎖的某個(gè)線程發(fā)出了(線程a對(duì)共享變量所做修改的)消息.
  • 線程b獲取一個(gè)鎖, 實(shí)質(zhì)上是線程b接收了之前某個(gè)線程發(fā)出的(在釋放這個(gè)鎖之前對(duì)共享變量所做修改的)消息.
  • 線程a釋放鎖, 隨后線程b獲取這個(gè)鎖, 這個(gè)過(guò)程實(shí)質(zhì)上是線程a通過(guò)主內(nèi)存向線程b發(fā)送消息.

3. 鎖內(nèi)存語(yǔ)義的實(shí)現(xiàn)

下面以reentrantlock為例, 獲取到鎖就是把state改為1(不考慮重入), 釋放鎖時(shí)改為0.

而加鎖的關(guān)鍵代碼就是

?
1
2
3
protected final boolean compareandsetstate(int expect, int update) {
 return unsafe.compareandswapint(this, stateoffset, expect, update);
}

該方法以原子操作的方式更新state變量, 本文把java的compareandset()方法簡(jiǎn)稱為cas. jdk文檔對(duì)該方法的說(shuō)明如下: 如果當(dāng)前狀態(tài)值等于預(yù)期值, 則以原子方式將同步狀態(tài)設(shè)置為給定的更新值. 此操作具有volatile讀和寫(xiě)的內(nèi)存語(yǔ)義.

這里我們分別從編譯器和處理器的角度來(lái)分析: cas如何同時(shí)具有volatile讀和volatile寫(xiě)的內(nèi)存語(yǔ)義.

我們知道, 編譯器不會(huì)對(duì)volatile讀與volatile讀后面的任意內(nèi)存操作重排序; 編譯器不會(huì)對(duì)volatile寫(xiě)與volatile寫(xiě)前面的任意內(nèi)存操作重排序. 組合這兩個(gè)條件, 意味著為了同時(shí)實(shí)現(xiàn)volatile讀和volatile寫(xiě)的內(nèi)存語(yǔ)義, 編譯器不能對(duì)cas與cas前面和后面的任意內(nèi)存操作重排序.

下面我們來(lái)分析在常見(jiàn)的intel x86處理器中, cas是如何同時(shí)具有volatile讀和volatile寫(xiě)的內(nèi)存語(yǔ)義的.

下面是sun.misc.unsafe類(lèi)的compareandswapint()方法的源代碼.

?
1
public final native boolean compareandswapint(object var1, long var2, int var4, int var5);

可以看到, 這是一個(gè)本地方法調(diào)用. 這個(gè)本地方法在openjdk中依次調(diào)用的c++代碼為: unsafe.cpp, atomic.cpp 和 atomic_windows_x86.inline.hpp. 這個(gè)本地方法的最終實(shí)現(xiàn)在openjdk的如下位置: openjdk-7-fcs-src-b147-
27_jun_2011\openjdk\hotspot\src\os_cpu\windows_x86\vm\atomic_windows_x86.inline.hpp(對(duì)應(yīng)于
windows操作系統(tǒng), x86處理器). 下面是對(duì)應(yīng)于intel x86處理器的源代碼的片段.

?
1
2
3
4
5
6
7
8
9
10
11
inline jint atomic::cmpxchg (jint exchange_value, volatile jint* dest, jint compare_value) {
 // alternative for interlockedcompareexchange
 int mp = os::is_mp();
 __asm {
  mov edx, dest
  mov ecx, exchange_value
  mov eax, compare_value
  lock_if_mp(mp)
  cmpxchg dword ptr [edx], ecx
 }
}

如上面源代碼所示, 程序會(huì)根據(jù)當(dāng)前處理器的類(lèi)型來(lái)決定是否為cmpxchg指令添加lock前綴. 如果程序是在多處理器上運(yùn)行, 就為cmpxchg指令加上lock前綴(lock cmpxchg). 反之, 如果程序是在單處理器上運(yùn)行, 就省略lock前綴(單處理器自身會(huì)維護(hù)單處理器內(nèi)的順序一致性, 不需要lock前綴提供的內(nèi)存屏障效果).

intel的手冊(cè)對(duì)lock前綴的說(shuō)明如下.

  • 確保對(duì)內(nèi)存的讀-改-寫(xiě)操作原子執(zhí)行. 在pentium及pentium之前的處理器中, 帶有l(wèi)ock前綴的指令在執(zhí)行期間會(huì)鎖住總線, 使得其他處理器暫時(shí)無(wú)法通過(guò)總線訪問(wèn)內(nèi)存. 很顯然, 這會(huì)帶來(lái)昂貴的開(kāi)銷(xiāo). 從pentium 4、intel xeon及p6處理器開(kāi)始, intel使用緩存鎖定(cache locking)
    來(lái)保證指令執(zhí)行的原子性. 緩存鎖定將大大降低lock前綴指令的執(zhí)行開(kāi)銷(xiāo).
  • 禁止該指令, 與之前和之后的讀和寫(xiě)指令重排序.
  • 把寫(xiě)緩沖區(qū)中的所有數(shù)據(jù)刷新到內(nèi)存中.

上面的第2點(diǎn)和第3點(diǎn)所具有的內(nèi)存屏障效果, 足以同時(shí)實(shí)現(xiàn)volatile讀和volatile寫(xiě)的內(nèi)存語(yǔ)義.

經(jīng)過(guò)上面的分析, 現(xiàn)在我們終于能明白為什么jdk文檔說(shuō)cas同時(shí)具有volatile讀和volatile寫(xiě)的內(nèi)存語(yǔ)義了.

從本文對(duì)reentrantlock的分析可以看出, 鎖釋放-獲取的內(nèi)存語(yǔ)義的實(shí)現(xiàn)至少有下面兩種方式.

  • 利用volatile變量的寫(xiě)-讀所具有的內(nèi)存語(yǔ)義.
  • 利用cas所附帶的volatile讀和volatile寫(xiě)的內(nèi)存語(yǔ)義.

4. 總結(jié)

對(duì)于鎖, 可以這么理解, n個(gè)線程去通過(guò)cas去修改一個(gè)volatile變量, 但是由于cpu提供的機(jī)制, 只能有一個(gè)線程修改成功, 修改成功的線程獲得鎖, 其它線程以及后來(lái)的線程要么自旋一會(huì)兒, 要么直接掛起, 等待獲取鎖的線程釋放鎖時(shí)去喚醒. 就是這么個(gè)過(guò)程.

好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)服務(wù)器之家的支持。

原文鏈接:https://www.cnblogs.com/wuqinglong/p/9962142.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 日本最新伦中文字幕 | 精品午夜中文字幕熟女人妻在线 | 美女脱一光二净的视频 | 亚洲国产综合网 | 国产成人一区二区三区影院免费 | 欧美va天堂| 成人精品一区二区三区 | 千金肉奴隶免费观看 | 欧美日韩成人在线视频 | 日本ssswww大学生 | 国产特黄a级在线视频 | 高h肉爽文农民工 | 无码精品AV久久久奶水 | 国产成人精品一区二区不卡 | 天码毛片一区二区三区入口 | 国产伦久视频免费观看视频 | 蘑菇香蕉茄子绿巨人丝瓜草莓 | 日本亚洲欧洲高清有码在线播放 | 国产精品原创巨作无遮挡 | 丰满大屁股美女一级毛片 | 男人的天堂在线观看入口 | 久久精品视频uu | 特黄特色大片免费影院 | 任我鲁精品视频精品 | 九九大香尹人视频免费 | 国产成人手机在线 | 国产福利不卡 | 秋霞黄色网 | 无码国产成人777爽死在线观看 | 日本破处 | 日本中文字幕一区二区高清在线 | av排名| 亚洲高清国产品国语在线观看 | 男人使劲躁女人小视频 | 亚洲精品123区在线观看 | k逼| 男女天堂 | 日产精品一卡2卡三卡4乱码久久 | 奇米影视一区 | 国产精品合集一区二区 | 小货SAO边洗澡边CAO你动漫 |