Compare and Swap [CAS] 算法 - Java 多線程

簡(jiǎn)書(shū) 賈小強(qiáng)
轉(zhuǎn)載請(qǐng)注明原創(chuàng)出處,謝謝!

一個(gè)Java 5中最好的補(bǔ)充是對(duì)原子操作的支持類(lèi)揣云,如AtomicIntegerAtomicLong等冰啃。這些類(lèi)幫助你減少?gòu)?fù)雜的(不必要的)多線程代碼邓夕,實(shí)際上只是完成一些基本操作刘莹,如增加或減少多個(gè)線程之間的共享的值。這些類(lèi)在內(nèi)部依賴(lài)于一個(gè)名為CAS(Compare and Swap)的算法焚刚。在這篇文章中点弯,我將詳細(xì)討論這個(gè)概念。

樂(lè)觀悲觀鎖 (Optimistic and Pessimistic Lock)

傳統(tǒng)的鎖機(jī)制汪榔,例如在Java中使用的synchronized關(guān)鍵字蒲拉,是多線程的悲觀鎖技術(shù)。它要求您首先保證沒(méi)有其他線程將在某些操作之間進(jìn)行干擾(即鎖定對(duì)象)痴腌。

這就像是說(shuō):“請(qǐng)先把門(mén)關(guān)上雌团,否則會(huì)有其他的騙子進(jìn)來(lái)整理你的東西”。

雖然上面的方法是安全的士聪,它確實(shí)有效锦援,但它在性能上對(duì)應(yīng)用程序造成了嚴(yán)重的影響。原因很簡(jiǎn)單剥悟,等待線程不能做任何事情灵寺,除非它們也有機(jī)會(huì)執(zhí)行被保護(hù)的操作。

實(shí)際上存在一種更有效性能更好的方式区岗,在本質(zhì)上是樂(lè)觀的略板。在這種方法中,您繼續(xù)進(jìn)行更新慈缔,希望您能在不受干擾的情況下完成叮称。這種方法依賴(lài)于碰撞檢測(cè),來(lái)確定是否更新時(shí)被其他方面的干擾藐鹤,在這種情況下瓤檐,操作失敗可以重試(或不)。

樂(lè)觀的方法就像一句老話:“獲得原諒比獲得許可更容易”娱节,在這里“更容易”意味著“更有效”挠蛉。

Compare and Swap是這種樂(lè)觀方法的一個(gè)很好例子,我們將在下面討論肄满。

Compare and Swap算法

該算法將某個(gè)內(nèi)存位置的內(nèi)容與給定值進(jìn)行比較(compare)谴古,只有當(dāng)它們相同時(shí),才將內(nèi)存位置的內(nèi)容修改為給定的新值稠歉。這是作為單個(gè)原子操作完成的讥电。原子性保證了新值是在最新值基礎(chǔ)上進(jìn)行計(jì)算的;如果該值已被另一個(gè)線程同時(shí)更新轧抗,那么修改會(huì)失敗。操作的結(jié)果必須指明它是否執(zhí)行替換(swap)瞬测,這可以通過(guò)簡(jiǎn)單的布爾響應(yīng)(這個(gè)變體通常稱(chēng)為compare-and-set)横媚,或者返回從內(nèi)存位置讀取的值(而不是寫(xiě)入它的值)來(lái)完成纠炮。

比如CAS操作有3個(gè)參數(shù):

  1. 一個(gè)內(nèi)存位置V,其值必須被替換灯蝴。
  2. 上一次線程讀的舊值A(chǔ)恢口。
  3. 應(yīng)在V上寫(xiě)的新值B。

CAS說(shuō):“我覺(jué)得V應(yīng)該有值A(chǔ)穷躁;如果是耕肩,把B放在那里,否則不要改變它问潭,但告訴我猿诸,我錯(cuò)了〗泼Γ”CAS樂(lè)觀的希望的更新成功梳虽,同時(shí)如果另一個(gè)線程從它上次檢查后更新了變量,它會(huì)檢測(cè)到失敗灾茁。

讓我們用一個(gè)例子明白整個(gè)過(guò)程窜觉。假設(shè)V是存儲(chǔ)值“10”的內(nèi)存位置。有多個(gè)線程希望增加這個(gè)值北专,并將遞增的值用于其他操作禀挫。讓我們逐步完成整個(gè)CAS操作:

  1. 初始狀態(tài)。
    V = 10, A = 0, B = 0
  2. 現(xiàn)在線程1首先出現(xiàn)拓颓,并將V與它的上次讀取的值A(chǔ)進(jìn)行比較(compare):
    V = 10, A = 10, B = 11
if A = V
   V = B
 else
   operation failed
   return V

顯然语婴,V的值將被替換(swap)為11,即操作成功录粱。

  1. 然后線程2執(zhí)行與線程1相同的操作腻格,這個(gè)時(shí)候內(nèi)置位置V實(shí)際上值已經(jīng)是11,但線程2上次讀取的還是10啥繁。
    V = 11, A = 10, B = 11
if A = V
   V = B
 else
   operation failed
   return V
  1. 在這種情況下菜职,將V與它的上次讀取的值A(chǔ)進(jìn)行比較(compare),V不等于A旗闽,所以不替換值酬核,返回V,而即當(dāng)前值11∈适遥現(xiàn)在線程2嫡意,再次用值重試(retry)這個(gè)操作:
    V = 11, A = 11, B = 12
    此時(shí),條件滿足并將值遞增為12捣辆,然后返回給線程2蔬螟。

總之,當(dāng)多個(gè)線程嘗試使用CAS同時(shí)更新同一個(gè)變量時(shí)汽畴,其中一個(gè)線程會(huì)成功更新變量的值旧巾,剩下的會(huì)失敗耸序。但失敗者不會(huì)受到線程的懲罰。他們可以自由地重試操作或干脆什么也不做鲁猩。

這樣每個(gè)線程對(duì)值得增加都是實(shí)打?qū)嵉脑黾恿丝补郑粫?huì)導(dǎo)致線程1,線程2同時(shí)讀取V為10廓握,然后線程1增加為11搅窿,線程2也自認(rèn)為增加了,但結(jié)果還是11隙券,通過(guò)CAS算法避免了消失的遞增男应,解決了明明線程2任務(wù)執(zhí)行了,計(jì)數(shù)器卻沒(méi)計(jì)數(shù)的悲劇**

這就是有關(guān)Java的原子(atomic)操作簡(jiǎn)單但重要的概念是尔。

Happy Learning !!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末殉了,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子拟枚,更是在濱河造成了極大的恐慌薪铜,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,284評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件恩溅,死亡現(xiàn)場(chǎng)離奇詭異隔箍,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)脚乡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)蜒滩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人奶稠,你說(shuō)我怎么就攤上這事俯艰。” “怎么了锌订?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,614評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵竹握,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我辆飘,道長(zhǎng)啦辐,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,671評(píng)論 1 293
  • 正文 為了忘掉前任蜈项,我火速辦了婚禮芹关,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘紧卒。我一直安慰自己侥衬,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,699評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著轴总,像睡著了一般贬媒。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上肘习,一...
    開(kāi)封第一講書(shū)人閱讀 51,562評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音坡倔,去河邊找鬼漂佩。 笑死,一個(gè)胖子當(dāng)著我的面吹牛罪塔,可吹牛的內(nèi)容都是我干的投蝉。 我是一名探鬼主播,決...
    沈念sama閱讀 40,309評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼征堪,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼瘩缆!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起佃蚜,我...
    開(kāi)封第一講書(shū)人閱讀 39,223評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤庸娱,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后谐算,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體熟尉,經(jīng)...
    沈念sama閱讀 45,668評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,859評(píng)論 3 336
  • 正文 我和宋清朗相戀三年洲脂,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了斤儿。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,981評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡恐锦,死狀恐怖往果,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情一铅,我是刑警寧澤陕贮,帶...
    沈念sama閱讀 35,705評(píng)論 5 347
  • 正文 年R本政府宣布,位于F島的核電站馅闽,受9級(jí)特大地震影響飘蚯,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜福也,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,310評(píng)論 3 330
  • 文/蒙蒙 一局骤、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧暴凑,春花似錦峦甩、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,904評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)犬辰。三九已至,卻和暖如春冰单,著一層夾襖步出監(jiān)牢的瞬間幌缝,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,023評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工诫欠, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留涵卵,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,146評(píng)論 3 370
  • 正文 我出身青樓荒叼,卻偏偏與公主長(zhǎng)得像轿偎,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子被廓,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,933評(píng)論 2 355

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