<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    qileilove

    blog已經(jīng)轉(zhuǎn)移至github,大家請(qǐng)?jiān)L問(wèn) http://qaseven.github.io/

    JAVA并發(fā)編程學(xué)習(xí)筆記之CAS操作

     CAS操作
      CAS是單詞compare and set的縮寫,意思是指在set之前先比較該值有沒(méi)有變化,只有在沒(méi)變的情況下才對(duì)其賦值。
      我們常常做這樣的操作
      if(a==b) {
      a++;
      }
      試想一下如果在做a++之前a的值被改變了怎么辦?a++還執(zhí)行嗎?出現(xiàn)該問(wèn)題的原因是在多線程環(huán)境下,a的值處于一種不定的狀態(tài)。采用鎖可以解決此類問(wèn)題,但CAS也可以解決,而且可以不加鎖。
    int expect = a;
    if(a.compareAndSet(expect,a+1)) {
    doSomeThing1();
    } else {
    doSomeThing2();
    }
      這樣如果a的值被改變了a++就不會(huì)被執(zhí)行。
      按照上面的寫法,a!=expect之后,a++就不會(huì)被執(zhí)行,如果我們還是想執(zhí)行a++操作怎么辦,沒(méi)關(guān)系,可以采用while循環(huán)
    while(true) {
    int expect = a;
    if (a.compareAndSet(expect, a + 1)) {
    doSomeThing1();
    return;
    } else {
    doSomeThing2();
    }
    }
      采用上面的寫法,在沒(méi)有鎖的情況下實(shí)現(xiàn)了a++操作,這實(shí)際上是一種非阻塞算法。
      應(yīng)用
       java.util.concurrent.atomic包中幾乎大部分類都采用了CAS操作,以AtomicInteger為例,看看它幾個(gè)主要方法的實(shí)現(xiàn):
    public final int getAndSet(int newValue) {
    for (;;) {
    int current = get();
    if (compareAndSet(current, newValue))
    return current;
    }
    }
      getAndSet方法JDK文檔中的解釋是:以原子方式設(shè)置為給定值,并返回舊值。原子方式體現(xiàn)在何處,就體現(xiàn)在compareAndSet上,看看compareAndSet是如何實(shí)現(xiàn)的:
      public final boolean compareAndSet(int expect, int update) {
      return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
      }
      不出所料,它就是采用的Unsafe類的CAS操作完成的。


      再來(lái)看看a++操作是如何實(shí)現(xiàn)的:
    public final int getAndIncrement() {
    for (;;) {
    int current = get();
    int next = current + 1;
    if (compareAndSet(current, next))
    return current;
    }
    }
      幾乎和最開始的實(shí)例一模一樣,也是采用CAS操作來(lái)實(shí)現(xiàn)自增操作的。
      ++a操作和a++操作類似,只不過(guò)返回結(jié)果不同罷了
    public final int incrementAndGet() {
    for (;;) {
    int current = get();
    int next = current + 1;
    if (compareAndSet(current, next))
    return next;
    }
    }
      此外,java.util.concurrent.ConcurrentLinkedQueue類全是采用的非阻塞算法,里面沒(méi)有使用任何鎖,全是基于CAS操作實(shí)現(xiàn)的。CAS操作可以說(shuō)是JAVA并發(fā)框架的基礎(chǔ),整個(gè)框架的設(shè)計(jì)都是基于CAS操作的。
      缺點(diǎn):
      1、ABA問(wèn)題
      CAS操作容易導(dǎo)致ABA問(wèn)題,也就是在做a++之間,a可能被多個(gè)線程修改過(guò)了,只不過(guò)回到了最初的值,這時(shí)CAS會(huì)認(rèn)為a的值沒(méi)有變。a在外面逛了一圈回來(lái),你能保證它沒(méi)有做任何壞事,不能!!也許它討閑,把b的值減了一下,把c的值加了一下等等,更有甚者如果a是一個(gè)對(duì)象,這個(gè)對(duì)象有可能是新創(chuàng)建出來(lái)的,a是一個(gè)引用呢情況又如何,所以這里面還是存在著很多問(wèn)題的,解決ABA問(wèn)題的方法有很多,可以考慮增加一個(gè)修改計(jì)數(shù),只有修改計(jì)數(shù)不變的且a值不變的情況下才做a++,也可以考慮引入版本號(hào),當(dāng)版本號(hào)相同時(shí)才做a++操作等,這和事務(wù)原子性處理有點(diǎn)類似!
      2、比較花費(fèi)CPU資源,即使沒(méi)有任何爭(zhēng)用也會(huì)做一些無(wú)用功。
      3、會(huì)增加程序測(cè)試的復(fù)雜度,稍不注意就會(huì)出現(xiàn)問(wèn)題。
      總結(jié):
      可以用CAS在無(wú)鎖的情況下實(shí)現(xiàn)原子操作,但要明確應(yīng)用場(chǎng)合,非常簡(jiǎn)單的操作且又不想引入鎖可以考慮使用CAS操作,當(dāng)想要非阻塞地完成某一操作也可以考慮CAS。不推薦在復(fù)雜操作中引入CAS,會(huì)使程序可讀性變差,且難以測(cè)試,同時(shí)會(huì)出現(xiàn)ABA問(wèn)題。

    posted on 2013-11-19 11:07 順其自然EVO 閱讀(219) 評(píng)論(0)  編輯  收藏


    只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


    網(wǎng)站導(dǎo)航:
     
    <2013年11月>
    272829303112
    3456789
    10111213141516
    17181920212223
    24252627282930
    1234567

    導(dǎo)航

    統(tǒng)計(jì)

    常用鏈接

    留言簿(55)

    隨筆分類

    隨筆檔案

    文章分類

    文章檔案

    搜索

    最新評(píng)論

    閱讀排行榜

    評(píng)論排行榜

    主站蜘蛛池模板: 亚洲另类精品xxxx人妖| 九九免费观看全部免费视频| 日本视频免费在线| 国产免费久久久久久无码| 亚洲理论在线观看| 亚洲AV中文无码乱人伦在线视色| 一级毛片**不卡免费播| 国产拍拍拍无码视频免费| 亚洲一区免费在线观看| 亚洲人成网站色在线入口 | 蜜桃精品免费久久久久影院| 国产伦精品一区二区免费| 亚洲午夜久久久久久尤物| 亚洲高清无码在线观看| 免费观看黄网站在线播放| 中文字幕av无码不卡免费| 中文无码亚洲精品字幕| 亚洲精品国产美女久久久| 日韩视频在线免费观看| 久久久免费精品re6| 五级黄18以上免费看| 国产亚洲精品VA片在线播放| 亚洲va久久久噜噜噜久久男同| 日韩免费高清视频| 免费观看激色视频网站bd| 十八禁在线观看视频播放免费| 亚洲av色香蕉一区二区三区 | 久久精品亚洲中文字幕无码网站| 女人18毛片水真多免费播放| 免费日本一区二区| 一级毛片不卡免费看老司机| 亚洲AV无码一区二区乱子仑| 亚洲第一区视频在线观看| 日韩亚洲一区二区三区| 国产成人精品久久亚洲| 日韩激情无码免费毛片| 中文字幕无码免费久久99| 日日麻批免费40分钟无码| 免费无码av片在线观看| 久青草视频97国内免费影视| 黄页免费视频播放在线播放|