JAVA并發(fā)編程(二):理解CAS機制

volatile_logo

也許大家已經聽說過官卡,鎖分兩種刻坊,一個叫悲觀鎖枷恕,一種稱之為樂觀鎖。Synchronized就是悲觀鎖的一種谭胚,也稱之為獨占鎖徐块,加了synchronized關鍵字的代碼基本上就只能以單線程的形式去執(zhí)行了,它會導致其他需要該資源的線程掛起灾而,直到前面的線程執(zhí)行完畢釋放所資源胡控。而另外一種樂觀鎖是一種更高效的機制,它的原理就是每次不加鎖去執(zhí)行某項操作旁趟,如果發(fā)生沖突則失敗并重試昼激,直到成功為止,其實本質上不算鎖锡搜,所以很多地方也稱之為自旋橙困。

一、并發(fā)編程中的原子性問題

在上篇JAVA并發(fā)編程(一):理解volatile關鍵字的結尾留了一個原子性的問題:

/*
 * i++ 的原子性問題:i++ 的操作實際上分為三個步驟“讀-改-寫”
 *        int i = 10;
 *        i = i++; //10
 * 
 *        int temp = i;
 *        i = i + 1;
 *        i = temp;
 */
public class TestAtomicDemo {

    public static void main(String[] args) {
        AtomicDemo ad = new AtomicDemo();
        
        for (int i = 0; i < 10; i++) {
            new Thread(ad).start();
        }
    }
    
}

class AtomicDemo implements Runnable{
    
    private volatile int serialNumber = 0;
    
    @Override
    public void run() {
        
        try {
            Thread.sleep(200);
        } catch (InterruptedException e) {
        }
        
        System.out.println(getSerialNumber());
    }
    
    public int getSerialNumber(){
        return serialNumber++;
    }
}

運行結果并不總是0-10耕餐。要想解決這個問題题禀,你可能會說加Synchronized同步鎖:加了同步鎖之后玻孟,serialNumber++操作變成了原子性操作,所以最終的輸出一定是0-9蓝谨,代碼實現(xiàn)了線程安全。

    public static void main(String[] args) {
        AtomicDemo ad = new AtomicDemo();
        
        for (int i = 0; i < 10; i++) {
            synchronized (TestAtomicDemo.class) {
                new Thread(ad).start();             
            }
        }
    }

但是眾所周知的是Synchronized同步鎖比較耗費性能,在某些情況下并不是一個好的選擇,那有沒有什么好的辦法呢?在JDK1.5之后壹蔓,java.util.concurrent.atomic包下為我們封裝了常用的原子變量,他們底層就是使用了CAS(compare and swap)算法來保證原子性猫态,我們將上面例子中的serialNumber改為使用AtomicInteger修飾佣蓉,然后運行發(fā)現(xiàn)也可以得到正確的結果。

//  private volatile int serialNumber = 0;
    
    private AtomicInteger serialNumber = new AtomicInteger(0);

二懂鸵、什么是CAS偏螺?

CAS是英文單詞Compare And Swap的縮寫,翻譯過來就是比較并替換匆光。
CAS機制當中使用了3個基本操作數(shù):內存地址V套像,舊的預期值A,要修改的新值B终息。
更新一個變量的時候夺巩,只有當變量的預期值A和內存地址V當中的實際值相同時,才會將內存地址V對應的值修改為B周崭。
來看一個例子:
我們現(xiàn)在有兩個線程:
1.在內存地址V當中柳譬,存儲著值為10的變量。
2.此時線程1想要把變量的值增加1续镇。對線程1來說美澳,舊的預期值A=10,要修改的新值B=11摸航。
3.在線程1要提交更新之前制跟,另一個線程2搶先一步,把內存地址V中的變量值率先更新成了11酱虎。
4.線程1開始提交更新雨膨,首先進行A和地址V的實際值比較(Compare),發(fā)現(xiàn)A不等于V的實際值读串,提交失敗聊记。
5.線程1重新獲取內存地址V的當前值,并重新計算想要修改的新值恢暖。此時對線程1來說排监,A=11,B=12杰捂。這個重新嘗試的過程被稱為自旋社露。
6.這一次比較幸運,沒有其他線程改變地址V的值琼娘。線程1進行Compare峭弟,發(fā)現(xiàn)A和地址V的實際值是相等的。
7.線程1進行SWAP脱拼,把地址V的值替換為B瞒瘸,也就是12。
從思想上來說熄浓,Synchronized屬于悲觀鎖情臭,悲觀地認為程序中的并發(fā)情況嚴重,所以嚴防死守赌蔑。CAS屬于樂觀鎖俯在,樂觀地認為程序中的并發(fā)情況不那么嚴重,所以讓線程不斷去嘗試更新娃惯。

三跷乐、CAS的缺點

CAS機制這么巧妙,是不是在任何地方都比同步鎖要好趾浅?然而并不是這樣的愕提,CAS機制有以下幾個問題:

  • CPU開銷較大
    在并發(fā)量比較高的情況下,如果許多線程反復嘗試更新某一個變量皿哨,卻又一直更新不成功浅侨,循環(huán)往復,會給CPU帶來很大的壓力证膨。
  • 不能保證代碼塊的原子性
    CAS機制所保證的只是一個變量的原子性操作如输,而不能保證整個代碼塊的原子性。比如需要保證3個變量共同進行原子性的更新央勒,就不得不使用Synchronized了不见。
  • ABA問題
    什么是ABA呢?簡單說就是一個值從A改成了B订歪,又從B改成了A脖祈。
    1.假設內存中有一個值為A的變量,存儲在地址V當中刷晋。
    2.此時有三個線程想使用CAS的方式更新這個變量值盖高,每個線程的執(zhí)行時間有略微的偏差。線程1和線程2已經獲得當前值眼虱,線程3還未獲得當前值喻奥。
    3.接下來,線程1先一步執(zhí)行成功捏悬,把當前值成功從A更新為B撞蚕;同時線程2因為某種原因被阻塞住,沒有做更新操作过牙;線程3在線程1更新之后甥厦,獲得了當前值B纺铭。
    4.再之后,線程2仍然處于阻塞狀態(tài)刀疙,線程3繼續(xù)執(zhí)行舶赔,成功把當前值從B更新成了A。
    5.最后谦秧,線程2終于恢復了運行狀態(tài)竟纳,由于阻塞之前已經獲得了“當前值”A,并且經過compare檢測疚鲤,內存地址V中的實際值也是A锥累,所以成功把變量值A更新成了B。
    6.這個過程中集歇,線程2獲取到的變量值A是一個舊值桶略,盡管和當前的實際值相同,但內存地址V中的變量已經經歷了A->B->A的改變鬼悠。
    從表面看起來運行結果好像沒什么問題删性,但是結合實際情況就會出現(xiàn)問題了。比如取款時有可能發(fā)生兩個線程同時扣款成功的情況焕窝。所以蹬挺,真正要做到嚴謹?shù)腃AS機制,我們在Compare階段不僅要比較期望值A和地址V中的實際值它掂,還要比較變量的版本號是否一致巴帮。
    在Java當中,AtomicStampedReference類就實現(xiàn)了用版本號做比較的CAS機制虐秋。

參考文章


本文作者: catalinaLi
本文鏈接: http://catalinali.top/2018/helloCAS/

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市客给,隨后出現(xiàn)的幾起案子用押,更是在濱河造成了極大的恐慌,老刑警劉巖靶剑,帶你破解...
    沈念sama閱讀 219,589評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蜻拨,死亡現(xiàn)場離奇詭異,居然都是意外死亡桩引,警方通過查閱死者的電腦和手機缎讼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,615評論 3 396
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來坑匠,“玉大人血崭,你說我怎么就攤上這事。” “怎么了夹纫?”我有些...
    開封第一講書人閱讀 165,933評論 0 356
  • 文/不壞的土叔 我叫張陵咽瓷,是天一觀的道長。 經常有香客問我捷凄,道長忱详,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,976評論 1 295
  • 正文 為了忘掉前任跺涤,我火速辦了婚禮,結果婚禮上监透,老公的妹妹穿的比我還像新娘桶错。我一直安慰自己,他們只是感情好胀蛮,可當我...
    茶點故事閱讀 67,999評論 6 393
  • 文/花漫 我一把揭開白布院刁。 她就那樣靜靜地躺著,像睡著了一般粪狼。 火紅的嫁衣襯著肌膚如雪退腥。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,775評論 1 307
  • 那天再榄,我揣著相機與錄音狡刘,去河邊找鬼。 笑死困鸥,一個胖子當著我的面吹牛嗅蔬,可吹牛的內容都是我干的。 我是一名探鬼主播疾就,決...
    沈念sama閱讀 40,474評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼澜术,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了猬腰?” 一聲冷哼從身側響起鸟废,我...
    開封第一講書人閱讀 39,359評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎姑荷,沒想到半個月后盒延,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 45,854評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡厢拭,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 38,007評論 3 338
  • 正文 我和宋清朗相戀三年兰英,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片供鸠。...
    茶點故事閱讀 40,146評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡畦贸,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情薄坏,我是刑警寧澤趋厉,帶...
    沈念sama閱讀 35,826評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站胶坠,受9級特大地震影響君账,放射性物質發(fā)生泄漏。R本人自食惡果不足惜沈善,卻給世界環(huán)境...
    茶點故事閱讀 41,484評論 3 331
  • 文/蒙蒙 一乡数、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧闻牡,春花似錦净赴、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,029評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至割以,卻和暖如春金度,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背严沥。 一陣腳步聲響...
    開封第一講書人閱讀 33,153評論 1 272
  • 我被黑心中介騙來泰國打工猜极, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人祝峻。 一個月前我還...
    沈念sama閱讀 48,420評論 3 373
  • 正文 我出身青樓魔吐,卻偏偏與公主長得像,于是被迫代替她去往敵國和親莱找。 傳聞我的和親對象是個殘疾皇子酬姆,可洞房花燭夜當晚...
    茶點故事閱讀 45,107評論 2 356

推薦閱讀更多精彩內容