volatile的應用

volatile是輕量級的synchronized,它在多處理器開發(fā)中保證了共享變量的“可見性”,它比synchronized的使用開銷低因為,他不會引起線程上下文的切換。

1.volatile的定義與實現(xiàn)原理

Java語言提供了volatile,在某些情況下比鎖要更方便。如果一個字段被聲明成volatile,Java線程內(nèi)存模型確保所有線程看到這個變量的值是一致的。

CPU指令:

在X86處理器下通過工具獲取JIT編譯器生成的匯編指令來查看對volatile進行寫操作時,CPU會做什么事情。

代碼:instance = new Singleton();? //instance是volatile變量。

匯編代碼: 0x01a3deld:movb $0X0,0X1104800(%esi);0x01a3de24:lock $0X0,(%esp);

有l(wèi)ock前綴在多核處理器下會引發(fā)兩件事。

1)將該處理器的該緩存行寫回到系統(tǒng)內(nèi)存中。

2)這個寫會內(nèi)存的操作會使其他處理器存儲了該地址數(shù)據(jù)的數(shù)據(jù)無效。

為了提高處理速度,處理器不直接和內(nèi)存進行通信,而是先將系統(tǒng)內(nèi)存的數(shù)據(jù)讀到內(nèi)部緩存(L1,L2或其他)后再進行操作,JVM就會向處理器發(fā)送一條Lock前綴的指令,將這個變量所在緩存行的數(shù)據(jù)寫回到系統(tǒng)內(nèi)存。但是,就算寫會到內(nèi)存,如果其他處理器緩存的值還是舊的,再執(zhí)行計算操作就會有問題。所以,在多處理器下,為了保證各個處理器的緩存是一致的,就會實現(xiàn)緩存一致性協(xié)議,每個處理器通過嗅探在總線上傳播的數(shù)據(jù)來檢查自己的緩存的值是不是過期了,當處理器發(fā)現(xiàn)自己緩存行對應的內(nèi)存地址被修改,就會將當前處理器的緩存行設(shè)置成無效狀態(tài),當處理器對這個數(shù)據(jù)進行修改操作的時候,會重新從系統(tǒng)內(nèi)存中把數(shù)據(jù)讀到處理器緩存里。

2.具體實現(xiàn)原則:

1)Lock前綴指令會引起處理器緩存回寫到內(nèi)存。Lock前綴指令導致在執(zhí)行指令期間,聲言處理器的LOCK#信號。在多處理器環(huán)境中,LOCK#信號確保在聲言該信號期間,處理器可以獨占任何共享內(nèi)存。但是在最近的處理器里,LOCK#信號一般不鎖總線,而是鎖緩存,畢竟鎖總線開銷的比較大。鎖緩存的原理是,當訪問的內(nèi)存區(qū)域已經(jīng)緩存在處理器內(nèi)部,處理器會鎖定這塊內(nèi)存區(qū)域的緩存行并回寫到內(nèi)存,并使用緩存一致性機制來確保修改的原子性,緩存一致性機制會阻止同時修改由兩個以上處理器緩存的內(nèi)存區(qū)域數(shù)據(jù)。

2)一個處理器的緩存回寫到內(nèi)存會導致其他處理器的緩存無效。處理器使用嗅探技術(shù)保證它的內(nèi)部緩存、系統(tǒng)內(nèi)存和其他處理器的緩存的數(shù)據(jù)在總線上保持一致。例如P6 family處理器中,如果通過嗅探一個處理器來檢測其他處理器打算寫內(nèi)存地址,而這個地址當前處于共享狀態(tài),那么正在嗅探的處理器將使它的緩存行無效,在下次訪問相同內(nèi)存地址時,強制執(zhí)行緩存行填充。


3.volatile的使用優(yōu)化

1)追加字節(jié)優(yōu)化性能:在JDK1.7的并發(fā)包里新增一個隊列集合類Linked-TransferQueue,它使用volatile變量時,用一種追加字節(jié)的方式來優(yōu)化隊列出隊和入隊的性能。

/**隊列中的頭部節(jié)點*/

private transient final PaddedAtomicReference<QNode> head;

/**隊列中的尾部節(jié)點*/

private transient final PaddedAtomicReference?tail;

static final class?PaddedAtomicReference <T> extends?AtomicReference <T>{

????//使用很多4個字節(jié)的引用追加到64個字節(jié)

? ? Object p0,p1,p2,p4,p5,p6,p7,p8,p9,pa,pb,pc,pd,pe;

? ??PaddedAtomicReference(T r){

? ? ? ? super(r);

????}

}

public class?AtomicReference <V> implements java.io.Serializable {

? ? private volatile V value;

? ? //省略其他代碼。

}

上述類,他使用一個內(nèi)部類類型來定義隊列的頭節(jié)點和尾節(jié)點,而這個內(nèi)部類相對于父類AtomicReference只做了一件事情,就是將共享變量追加到64字節(jié)。因為對于很多的處理器來說,他們的高速緩存行L1、L2和L3是64字節(jié)寬,不支持部分填充緩存行。這意味著隊列的頭節(jié)點和尾節(jié)點都不足64字節(jié)的話,處理器會將它們都讀到同一個高速緩存行中,再多處理器下每個處理器都會緩存同樣的頭節(jié)點和尾節(jié)點,當一個處理器試圖修改頭節(jié)點時,會將整個緩存行鎖定,那么在緩存一致性機制的影響下,其他的處理器將不能訪問它們的尾節(jié)點,而隊列的入隊列和出隊列操作則需要不停的修改頭節(jié)點和尾節(jié)點,所以在多處理的情況下將會嚴重影響到隊列的入隊和出隊效率。追加到64字節(jié)的方式來填滿告訴緩沖區(qū)的緩存行,避免頭節(jié)點和尾節(jié)點加載到同一個緩存行,使頭、尾節(jié)點在修改時不會互相鎖定。

如果共享變量不被頻繁寫的話,鎖的幾率也非常小,就沒必要通過追加字節(jié)的方式來避免相互鎖定。

但是這種追加字節(jié)的方式在Java7下可能不生效,因為Java7變得更加智慧,他會淘汰或重新排列無用字段,需要使用其他追加字節(jié)的方式。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,106評論 6 542
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,441評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,211評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,736評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,475評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,834評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,829評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,009評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,559評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,306評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,516評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,038評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,728評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,132評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,443評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,249評論 3 399
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,484評論 2 379

推薦閱讀更多精彩內(nèi)容

  • 從三月份找實習到現(xiàn)在,面了一些公司,掛了不少,但最終還是拿到小米、百度、阿里、京東、新浪、CVTE、樂視家的研發(fā)崗...
    時芥藍閱讀 42,333評論 11 349
  • volatile關(guān)鍵字 當變量被某個線程A修改值之后,其它線程比如B若讀取此變量的話,立刻可以看到原來線程A修改后...
    生活理當如此閱讀 542評論 0 0
  • 一個簡單的單例示例 單例模式可能是大家經(jīng)常接觸和使用的一個設(shè)計模式,你可能會這么寫 publicclassUnsa...
    Martin說閱讀 2,257評論 0 6
  • 你信么,蝸牛可以待在殼里一年不出來。 我好像是只蝸牛,出生后頭上幾根毛發(fā),皮膚皺的像沙皮狗,而且還不會...
    站在地面飛閱讀 488評論 0 2
  • 0327 楊:打卡,《極盜者》第二遍。等這個電影的清晰版等了快四個月。電影的主要內(nèi)容是人與自然的關(guān)系,與自然融為一...
    鹿_NICE閱讀 203評論 0 0