1. 概述
鎖是Java并發(fā)編程中最重要的同步機(jī)制。鎖除了讓臨界區(qū)互斥執(zhí)行外,還可以讓釋放鎖的線(xiàn)程獲取同一個(gè)鎖的線(xiàn)程發(fā)送消息。
鎖在實(shí)際使用時(shí)只是明白鎖限制了并發(fā)訪(fǎng)問(wèn), 但是鎖是如何實(shí)現(xiàn)并發(fā)訪(fǎng)問(wèn)的, 同學(xué)們可能不太清楚, 下面這篇文章就來(lái)揭開(kāi)鎖的神秘面紗.
2. 鎖的內(nèi)存語(yǔ)義
鎖的內(nèi)存語(yǔ)義與volatile的內(nèi)存語(yǔ)義
內(nèi)存語(yǔ)義總結(jié)
3. 鎖內(nèi)存語(yǔ)義的實(shí)現(xiàn)
下面以ReentrantLock為例, 獲取到鎖就是把state改為1(不考慮重入), 釋放鎖時(shí)改為0.
而加鎖的關(guān)鍵代碼就是
protected final boolean compareAndSetState(int expect, int update) { return unsafe.compareAndSwapInt(this, stateOffset, expect, update);}
該方法以原子操作的方式更新state變量, 本文把Java的compareAndSet()方法簡(jiǎn)稱(chēng)為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()方法的源代碼.
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處理器的源代碼的片段.
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ō)明如下.
上面的第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)至少有下面兩種方式.
4. 總結(jié)
對(duì)于鎖, 可以這么理解, N個(gè)線(xiàn)程去通過(guò)CAS去修改一個(gè)volatile變量, 但是由于CPU提供的機(jī)制, 只能有一個(gè)線(xiàn)程修改成功, 修改成功的線(xiàn)程獲得鎖, 其它線(xiàn)程以及后來(lái)的線(xiàn)程要么自旋一會(huì)兒, 要么直接掛起, 等待獲取鎖的線(xiàn)程釋放鎖時(shí)去喚醒. 就是這么個(gè)過(guò)程.
好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)VeVb武林網(wǎng)的支持。
|
新聞熱點(diǎn)
疑難解答
圖片精選