4.CAS

CAS

在JDK 5之前Java語(yǔ)言是靠synchronized關(guān)鍵字保證同步的茴迁,這會(huì)導(dǎo)致有鎖(后面的章節(jié)還會(huì)談到鎖)。

鎖機(jī)制存在以下問(wèn)題:

(1)在多線程競(jìng)爭(zhēng)下俺亮,加鎖沥邻、釋放鎖會(huì)導(dǎo)致比較多的上下文切換和調(diào)度延時(shí)平项,引起性能問(wèn)題赫舒。

(2)一個(gè)線程持有鎖會(huì)導(dǎo)致其它所有需要此鎖的線程掛起。

(3)如果一個(gè)優(yōu)先級(jí)高的線程等待一個(gè)優(yōu)先級(jí)低的線程釋放鎖會(huì)導(dǎo)致優(yōu)先級(jí)倒置闽瓢,引起性能風(fēng)險(xiǎn)接癌。

volatile是不錯(cuò)的機(jī)制,但是volatile不能保證原子性扣讼。因此對(duì)于同步最終還是要回到鎖機(jī)制上來(lái)缺猛。

獨(dú)占鎖是一種悲觀鎖,synchronized就是一種獨(dú)占鎖,會(huì)導(dǎo)致其它所有需要鎖的線程掛起荔燎,等待持有鎖的線程釋放鎖耻姥。而另一個(gè)更加有效的鎖就是樂(lè)觀鎖。所謂樂(lè)觀鎖就是有咨,每次不加鎖而是假設(shè)沒(méi)有沖突而去完成某項(xiàng)操作咏闪,如果因?yàn)闆_突失敗就重試,直到成功為止摔吏。

CAS 操作

上面的樂(lè)觀鎖用到的機(jī)制就是CAS,Compare and Swap纵装。

CAS有3個(gè)操作數(shù)征讲,內(nèi)存值V,舊的預(yù)期值A(chǔ)橡娄,要修改的新值B诗箍。當(dāng)且僅當(dāng)預(yù)期值A(chǔ)和內(nèi)存值V相同時(shí),將內(nèi)存值V修改為B挽唉,否則什么都不做滤祖。

非阻塞算法 (nonblocking algorithms)

一個(gè)線程的失敗或者掛起不應(yīng)該影響其他線程的失敗或掛起的算法。

現(xiàn)代的CPU提供了特殊的指令瓶籽,可以自動(dòng)更新共享數(shù)據(jù)匠童,而且能夠檢測(cè)到其他線程的干擾,而 compareAndSet() 就用這些代替了鎖定塑顺。

拿出AtomicInteger來(lái)研究在沒(méi)有鎖的情況下是如何做到數(shù)據(jù)正確性的汤求。

private volatile int value;

首先毫無(wú)以為,在沒(méi)有鎖的機(jī)制下可能需要借助volatile原語(yǔ)严拒,保證線程間的數(shù)據(jù)是可見(jiàn)的(共享的)扬绪。

這樣才獲取變量的值的時(shí)候才能直接讀取。

public final int get() {
? return value;
? }

然后來(lái)看看++i是怎么做到的裤唠。

public final int incrementAndGet() {
? for (;;) {
? int current = get();
? int next = current + 1;
? if (compareAndSet(current, next))
? return next;
? }
}

在這里采用了CAS操作挤牛,每次從內(nèi)存中讀取數(shù)據(jù)然后將此數(shù)據(jù)和+1后的結(jié)果進(jìn)行CAS操作,如果成功就返回結(jié)果种蘸,否則重試直到成功為止墓赴。

而compareAndSet利用JNI來(lái)完成CPU指令的操作。

public final boolean compareAndSet(int expect, int update) {
? return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
? }

整體的過(guò)程就是這樣子的劈彪,利用CPU的CAS指令竣蹦,同時(shí)借助JNI來(lái)完成Java的非阻塞算法。其它原子操作都是利用類似的特性完成的沧奴。

而整個(gè)J.U.C都是建立在CAS之上的痘括,因此對(duì)于synchronized阻塞算法,J.U.C在性能上有了很大的提升。參考資料的文章中介紹了如果利用CAS構(gòu)建非阻塞計(jì)數(shù)器纲菌、隊(duì)列等數(shù)據(jù)結(jié)構(gòu)挠日。

CAS看起來(lái)很爽,但是會(huì)導(dǎo)致“ABA問(wèn)題”翰舌。

CAS算法實(shí)現(xiàn)一個(gè)重要前提需要取出內(nèi)存中某時(shí)刻的數(shù)據(jù)嚣潜,而在下時(shí)刻比較并替換,那么在這個(gè)時(shí)間差類會(huì)導(dǎo)致數(shù)據(jù)的變化椅贱。

比如說(shuō)一個(gè)線程one從內(nèi)存位置V中取出A懂算,這時(shí)候另一個(gè)線程two也從內(nèi)存中取出A,并且two進(jìn)行了一些操作變成了B庇麦,然后two又將V位置的數(shù)據(jù)變成A计技,這時(shí)候線程one進(jìn)行CAS操作發(fā)現(xiàn)內(nèi)存中仍然是A,然后one操作成功山橄。盡管線程one的CAS操作成功垮媒,但是不代表這個(gè)過(guò)程就是沒(méi)有問(wèn)題的。如果鏈表的頭在變化了兩次后恢復(fù)了原值航棱,但是不代表鏈表就沒(méi)有變化睡雇。因此前面提到的原子操作AtomicStampedReference/AtomicMarkableReference就很有用了。這允許一對(duì)變化的元素進(jìn)行原子操作饮醇。

參考資料:

(1)非阻塞算法簡(jiǎn)介

(2)流行的原子

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末它抱,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子朴艰,更是在濱河造成了極大的恐慌抗愁,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,755評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件呵晚,死亡現(xiàn)場(chǎng)離奇詭異蜘腌,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)饵隙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)撮珠,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人金矛,你說(shuō)我怎么就攤上這事芯急。” “怎么了驶俊?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,138評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵娶耍,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我饼酿,道長(zhǎng)榕酒,這世上最難降的妖魔是什么胚膊? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,791評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮想鹰,結(jié)果婚禮上紊婉,老公的妹妹穿的比我還像新娘。我一直安慰自己辑舷,他們只是感情好喻犁,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著何缓,像睡著了一般肢础。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上碌廓,一...
    開(kāi)封第一講書(shū)人閱讀 51,631評(píng)論 1 305
  • 那天乔妈,我揣著相機(jī)與錄音,去河邊找鬼氓皱。 笑死,一個(gè)胖子當(dāng)著我的面吹牛勃刨,可吹牛的內(nèi)容都是我干的波材。 我是一名探鬼主播,決...
    沈念sama閱讀 40,362評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼身隐,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼廷区!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起贾铝,我...
    開(kāi)封第一講書(shū)人閱讀 39,264評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤隙轻,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后垢揩,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體玖绿,經(jīng)...
    沈念sama閱讀 45,724評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年叁巨,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了斑匪。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,040評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡锋勺,死狀恐怖蚀瘸,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情庶橱,我是刑警寧澤贮勃,帶...
    沈念sama閱讀 35,742評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站苏章,受9級(jí)特大地震影響寂嘉,放射性物質(zhì)發(fā)生泄漏奏瞬。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評(píng)論 3 330
  • 文/蒙蒙 一垫释、第九天 我趴在偏房一處隱蔽的房頂上張望丝格。 院中可真熱鬧,春花似錦棵譬、人聲如沸显蝌。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,944評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)曼尊。三九已至,卻和暖如春脏嚷,著一層夾襖步出監(jiān)牢的瞬間骆撇,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,060評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工父叙, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留神郊,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,247評(píng)論 3 371
  • 正文 我出身青樓趾唱,卻偏偏與公主長(zhǎng)得像涌乳,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子甜癞,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評(píng)論 2 355

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