java 多個線程可以讀一個變量,只有一個線程可以對這個變量進行寫,到底要不要加鎖?
多個線程可以讀一個變量,只有一個線程可以對這個變量進行寫,到底要不要加鎖?下面簡要解釋一下原因:鎖定是因為操作不是原子的。讓我們用操作一來解釋它??聪旅鎯蓚€圖。我這個操作需要看上面的第二個圖,你能很清
多個線程可以讀一個變量,只有一個線程可以對這個變量進行寫,到底要不要加鎖?
下面簡要解釋一下原因:
鎖定是因為操作不是原子的。讓我們用操作一來解釋它。看下面兩個圖。
我這個操作需要
看上面的第二個圖,你能很清楚地理解這個過程嗎?
鎖定是為了確保上述三個步驟是原子操作。
回到問題上來,只有一個線程要寫,沒有競爭,所以不需要鎖定。
但是,如果你看第一張圖片,因為主內(nèi)存和本地內(nèi)存的存在
在一個線程寫入后,其他線程無法立即看到它。這就是可見性問題。
添加volatile關(guān)鍵字后,它將在操作后強制工作內(nèi)存和主內(nèi)存同步,以確保其他線程可以立即看到它。
volatile關(guān)鍵字在Java中有什么作用?
Volatile是為了防止指令重新排序以確保可見性
對于JVM級別,是為了防止編譯器重新排序
同時,對于某些CPU,它們會通過緩存鎖或線程來解決緩存可見性
但是,目前很多CPU已經(jīng)過優(yōu)化,由于cache一致性MESI會帶來性能開銷,因此采用storebuffer機制進行異步處理,這種機制會導(dǎo)致指令的無序執(zhí)行。這會導(dǎo)致可見性問題。
Volatile將在CPU級別增加內(nèi)存屏障,以解決CPU執(zhí)行無序?qū)е碌目梢娦詥栴}
共享鎖(s lock)也稱為讀鎖。如果事務(wù)t將s鎖添加到數(shù)據(jù)對象a,則事務(wù)t可以讀取a,但不能修改a。其他事務(wù)只能將s鎖添加到a,而不能添加x鎖,直到t釋放a上的s鎖。這確保了其他事務(wù)可以讀取a,但在t釋放a上的s鎖之前,t不能對a進行任何更改。
獨占鎖(x鎖)也稱為寫入鎖上。如果事務(wù)t將x鎖應(yīng)用于數(shù)據(jù)對象a,則事務(wù)t可以讀取a或修改a。在t釋放a上的鎖之前,其他事務(wù)無法將任何鎖應(yīng)用于a。這確保了在t釋放a上的鎖之前,其他事務(wù)無法讀取和修改a