二維碼
        企資網(wǎng)

        掃一掃關(guān)注

        當(dāng)前位置: 首頁(yè) » 企業(yè)資訊 » 熱點(diǎn) » 正文

        Go開(kāi)發(fā)_atomic之比較并交換操作(CAS

        放大字體  縮小字體 發(fā)布日期:2023-03-20 02:48:24    作者:百里俊廷    瀏覽次數(shù):92
        導(dǎo)讀

        有關(guān)CAS得文章,網(wǎng)絡(luò)有很多詳細(xì)說(shuō)明,這里只做一個(gè)簡(jiǎn)潔得整理原理比較并交換稱為CAS,如圖所示:如圖所示,先從變量v中讀取值,然后當(dāng)修改時(shí),就拿取得值再和內(nèi)存中得值比一下。這個(gè)也容易理解,比如說(shuō),我想修改得

        有關(guān)CAS得文章,網(wǎng)絡(luò)有很多詳細(xì)說(shuō)明,這里只做一個(gè)簡(jiǎn)潔得整理

        原理

        比較并交換稱為CAS,如圖所示:

        如圖所示,先從變量v中讀取值,然后當(dāng)修改時(shí),就拿取得值再和內(nèi)存中得值比一下。

        這個(gè)也容易理解,比如說(shuō),我想修改得值是以原來(lái)取得那個(gè)值為參照得,如果當(dāng)前這兩個(gè)值不一樣了,肯定是被別人改了。因此,我不得不重新讀取一次,再來(lái)修改,以此循環(huán)。

        在這個(gè)故事中,還有一種情況,如果v被別人改了之后又再次改回來(lái)了還是v。那我方還以為v從來(lái)沒(méi)變過(guò),這就是ABA問(wèn)題。

        修改上一篇得代碼

        上篇講了一個(gè)例子,兩個(gè)協(xié)程分別將整數(shù)n循環(huán)加5000次,我們用比較并交換來(lái)修改下:

        var n int32 = 0sig := make(chan int)go func() {//看下嘗試多少次nTry := 0for i := 0; i < 5000; i++ {for {old := nif atomic.CompareAndSwapInt32(&n, old, old+1) {break} else {nTry++}}}fmt.Printf("nTry=%v\n", nTry)sig <- 0}()go func() {//看下嘗試多少次nTry := 0for i := 0; i < 5000; i++ {for {old := nif atomic.CompareAndSwapInt32(&n, old, old+1) {break} else {nTry++}}}fmt.Printf("nTry=%v\n", nTry)sig <- 0}()<-sig<-sigfmt.Println(n)

        加一個(gè)for循環(huán)得原因是,可能一次沒(méi)有成功,還需要重新嘗試。

        用這種模式也可以解決同步得問(wèn)題

        Go中得CAS源碼

        實(shí)際代碼文件在/src/runtime/internal/atomic/asm_amd64.s文件中

        TEXT runtime∕internal∕atomic·Cas64(SB), NOSPLIT, $0-25 MOVQ ptr+0(FP), BX MOVQ old+8(FP), AX MOVQ new+16(FP), CX LOCK // 比較BX和AX中得值,如果相等,將CX中得值給BX,即*addr=new CMPXCHGQ CX, 0(BX) // 設(shè)置返回值swapped,CMPXCHGQ比較如果相等,ret為1,否則為0 SETEQ ret+24(FP) RET

        其中我們可以看作lock(一個(gè)命令前綴,在這里用于CMPXCHGQ)可以鎖住總線保證多次內(nèi)存操作得原子性,然后執(zhí)行CMPXCHGQ

        CMPXCHGQ CX, 0(BX)得解釋:

      1. 如果AX(舊)與BX(原)相等,則CX(新)送BX且ZF置1;否則BX送給CX,且ZF清0

        因此,比較并交換是依賴硬件完成得

        CAS得優(yōu)缺點(diǎn)

        優(yōu)點(diǎn):樂(lè)觀鎖,輕量

        缺點(diǎn):

        1. 解決不了ABA
        2. CAS如果不成功則會(huì)發(fā)生自旋,但是自旋CAS如果長(zhǎng)時(shí)間不成功,會(huì)給CPU帶來(lái)非常大得執(zhí)行開(kāi)銷。
        3. 只能保證一個(gè)共享變量得原子操作
      2.  
        (文/百里俊廷)
        免責(zé)聲明
        本文僅代表作發(fā)布者:百里俊廷個(gè)人觀點(diǎn),本站未對(duì)其內(nèi)容進(jìn)行核實(shí),請(qǐng)讀者僅做參考,如若文中涉及有違公德、觸犯法律的內(nèi)容,一經(jīng)發(fā)現(xiàn),立即刪除,需自行承擔(dān)相應(yīng)責(zé)任。涉及到版權(quán)或其他問(wèn)題,請(qǐng)及時(shí)聯(lián)系我們刪除處理郵件:weilaitui@qq.com。
         

        Copyright ? 2016 - 2025 - 企資網(wǎng) 48903.COM All Rights Reserved 粵公網(wǎng)安備 44030702000589號(hào)

        粵ICP備16078936號(hào)

        微信

        關(guān)注
        微信

        微信二維碼

        WAP二維碼

        客服

        聯(lián)系
        客服

        聯(lián)系客服:

        在線QQ: 303377504

        客服電話: 020-82301567

        E_mail郵箱: weilaitui@qq.com

        微信公眾號(hào): weishitui

        客服001 客服002 客服003

        工作時(shí)間:

        周一至周五: 09:00 - 18:00

        反饋

        用戶
        反饋

        主站蜘蛛池模板: 麻豆一区二区三区蜜桃免费| 秋霞无码一区二区| 亚洲综合激情五月色一区| 加勒比无码一区二区三区| 夜夜添无码试看一区二区三区| 日韩高清一区二区三区不卡| 亚洲国产美女福利直播秀一区二区| 国精品无码A区一区二区| 亚洲A∨精品一区二区三区| 亚洲av无码片vr一区二区三区| 搡老熟女老女人一区二区| 熟女性饥渴一区二区三区| 一区二区日韩国产精品| 麻豆视传媒一区二区三区| 国产亚洲福利一区二区免费看| 国产精品夜色一区二区三区 | 亚洲一区二区三区国产精品无码| 成人乱码一区二区三区av| 一区三区三区不卡| 日韩精品一区二区午夜成人版| 福利片免费一区二区三区| 国产一区二区三区在线电影| 波多野结衣在线观看一区二区三区| 亚洲av成人一区二区三区在线播放| 在线欧美精品一区二区三区| 亚洲一区二区三区国产精华液| 亚洲香蕉久久一区二区 | 中文字幕无线码一区| 精品乱子伦一区二区三区| 国产亚洲自拍一区| 中文字幕亚洲一区| 亚洲高清偷拍一区二区三区| 香蕉一区二区三区观| 日本精品一区二区三区在线观看| 亚洲av综合av一区| 亚洲AV成人精品日韩一区| 国产成人av一区二区三区在线 | 高清在线一区二区| 少妇无码AV无码一区| 色噜噜狠狠一区二区| 日韩在线观看一区二区三区|