青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

loop_in_codes

低調做技術__歡迎移步我的獨立博客 codemaro.com 微博 kevinlynx

無鎖有序鏈表的實現(xiàn)

無鎖有序鏈表可以保證元素的唯一性,使其可用于哈希表的桶,甚至直接作為一個效率不那么高的map。普通鏈表的無鎖實現(xiàn)相對簡單點,因為插入元素可以在表頭插,而有序鏈表的插入則是任意位置。

本文主要基于論文High Performance Dynamic Lock-Free Hash Tables實現(xiàn)。

主要問題

鏈表的主要操作包含insertremove,先簡單實現(xiàn)一個版本,就會看到問題所在,以下代碼只用作示例:

struct node_t {
        key_t key;
        value_t val;
        node_t *next;
    };

    int l_find(node_t **pred_ptr, node_t **item_ptr, node_t *head, key_t key) {
        node_t *pred = head;
        node_t *item = head->next;
        while (item) {
            int d = KEY_CMP(item->key, key);
            if (d >= 0) {
                *pred_ptr = pred;
                *item_ptr = item;
                return d == 0 ? TRUE : FALSE;
            }
            pred = item;
            item = item->next;
        } 
        *pred_ptr = pred;
        *item_ptr = NULL;
        return FALSE;
    }

    int l_insert(node_t *head, key_t key, value_t val) {
        node_t *pred, *item, *new_item;
        while (TRUE) {
            if (l_find(&pred, &item, head, key)) {
                return FALSE;
            }
            new_item = (node_t*) malloc(sizeof(node_t));
            new_item->key = key;
            new_item->val = val;
            new_item->next = item;
            // A. 如果pred本身被移除了
            if (CAS(&pred->next, item, new_item)) {
                return TRUE;
            }
            free(new_item);
        }
    }

    int l_remove(node_t *head, key_t key) {
        node_t *pred, *item;
        while (TRUE) {
            if (!l_find(&pred, &item, head, key)) {
                return TRUE;
            }
            // B. 如果pred被移除;如果item也被移除
            if (CAS(&pred->next, item, item->next)) {
                haz_free(item);
                return TRUE;
            }
        }
    }

l_find函數(shù)返回查找到的前序元素和元素本身,代碼A和B雖然拿到了preditem,但在CAS的時候,其可能被其他線程移除。甚至,在l_find過程中,其每一個元素都可能被移除。問題在于,任何時候拿到一個元素時,都不確定其是否還有效。元素的有效性包括其是否還在鏈表中,其指向的內存是否還有效。

解決方案

通過為元素指針增加一個有效性標志位,配合CAS操作的互斥性,就可以解決元素有效性判定問題。

因為node_t放在內存中是會對齊的,所以指向node_t的指針值低幾位是不會用到的,從而可以在低幾位里設置標志,這樣在做CAS的時候,就實現(xiàn)了DCAS的效果,相當于將兩個邏輯上的操作變成了一個原子操作。想象下引用計數(shù)對象的線程安全性,其內包裝的指針是線程安全的,但對象本身不是。

CAS的互斥性,在若干個線程CAS相同的對象時,只有一個線程會成功,失敗的線程就可以以此判定目標對象發(fā)生了變更。改進后的代碼(代碼僅做示例用,不保證正確):

typedef size_t markable_t;
    // 最低位置1,表示元素被刪除
    #define HAS_MARK(p) ((markable_t)p & 0x01)
    #define MARK(p) ((markable_t)p | 0x01)
    #define STRIP_MARK(p) ((markable_t)p & ~0x01)

    int l_insert(node_t *head, key_t key, value_t val) {
        node_t *pred, *item, *new_item;
        while (TRUE) {
            if (l_find(&pred, &item, head, key)) { 
                return FALSE;
            }
            new_item = (node_t*) malloc(sizeof(node_t));
            new_item->key = key;
            new_item->val = val;
            new_item->next = item;
            // A. 雖然find拿到了合法的pred,但是在以下代碼之前pred可能被刪除,此時pred->next被標記
            //    pred->next != item,該CAS會失敗,失敗后重試
            if (CAS(&pred->next, item, new_item)) {
                return TRUE;
            }
            free(new_item);
        }
        return FALSE;
    }

    int l_remove(node_t *head, key_t key) {
        node_t *pred, *item;
        while (TRUE) {
            if (!l_find(&pred, &item, head, key)) {
                return FALSE;
            }
            node_t *inext = item->next;
            // B. 刪除item前先標記item->next,如果CAS失敗,那么情況同insert一樣,有其他線程在find之后
            //    刪除了item,失敗后重試
            if (!CAS(&item->next, inext, MARK(inext))) {
                continue;
            }
            // C. 對同一個元素item刪除時,只會有一個線程成功走到這里
            if (CAS(&pred->next, item, STRIP_MARK(item->next))) {
                haz_defer_free(item);
                return TRUE;
            }
        }
        return FALSE;
    }

    int l_find(node_t **pred_ptr, node_t **item_ptr, node_t *head, key_t key) {
        node_t *pred = head;
        node_t *item = head->next;
        hazard_t *hp1 = haz_get(0);
        hazard_t *hp2 = haz_get(1);
        while (item) {
            haz_set_ptr(hp1, pred);
            haz_set_ptr(hp2, item);
            /* 
             如果已被標記,那么緊接著item可能被移除鏈表甚至釋放,所以需要重頭查找
            */
            if (HAS_MARK(item->next)) { 
                return l_find(pred_ptr, item_ptr, head, key);
            }
            int d = KEY_CMP(item->key, key);
            if (d >= 0) {
                *pred_ptr = pred;
                *item_ptr = item;
                return d == 0 ? TRUE : FALSE;
            }
            pred = item;
            item = item->next;
        } 
        *pred_ptr = pred;
        *item_ptr = NULL;
        return FALSE;
    }

haz_gethaz_set_ptr之類的函數(shù)是一個hazard pointer實現(xiàn),用于支持多線程下內存的GC。上面的代碼中,要刪除一個元素item時,會標記item->next,從而使得insert時中那個CAS不需要做任何調整。總結下這里的線程競爭情況:

  • insertfind到正常的preditem,pred->next == item,然后在CAS前有線程刪除了pred,此時pred->next == MARK(item),CAS失敗,重試;刪除分為2種情況:a) 從鏈表移除,得到標記,pred可繼續(xù)訪問;b) pred可能被釋放內存,此時再使用pred會錯誤。為了處理情況b,所以引入了類似hazard pointer的機制,可以有效保障任意一個指針p只要還有線程在使用它,它的內存就不會被真正釋放
  • insert中有多個線程在pred后插入元素,此時同樣由insert中的CAS保證,這個不多說
  • remove中情況同insert,find拿到了有效的prednext,但在CAS的時候pred被其他線程刪除,此時情況同insert,CAS失敗,重試
  • 任何時候改變鏈表結構時,無論是remove還是insert,都需要重試該操作
  • find中遍歷時,可能會遇到被標記刪除的item,此時item根據(jù)remove的實現(xiàn)很可能被刪除,所以需要重頭開始遍歷

ABA問題

ABA問題還是存在的,insert中:

if (CAS(&pred->next, item, new_item)) {
        return TRUE;
    }

如果CAS之前,pred后的item被移除,又以相同的地址值加進來,但其value變了,此時CAS會成功,但鏈表可能就不是有序的了。pred->val < new_item->val > item->val

為了解決這個問題,可以利用指針值地址對齊的其他位來存儲一個計數(shù),用于表示pred->next的改變次數(shù)。當insert拿到pred時,pred->next中存儲的計數(shù)假設是0,CAS之前其他線程移除了pred->next又新增回了item,此時pred->next中的計數(shù)增加,從而導致insertCAS失敗。

// 最低位留作刪除標志
    #define MASK ((sizeof(node_t) - 1) & ~0x01)

    #define GET_TAG(p) ((markable_t)p & MASK)
    #define TAG(p, tag) ((markable_t)p | (tag))
    #define MARK(p) ((markable_t)p | 0x01)
    #define HAS_MARK(p) ((markable_t)p & 0x01)
    #define STRIP_MARK(p) ((node_t*)((markable_t)p & ~(MASK | 0x01)))

remove的實現(xiàn):

/* 先標記再刪除 */
    if (!CAS(&sitem->next, inext, MARK(inext))) {
        continue;
    }
    int tag = GET_TAG(pred->next) + 1;
    if (CAS(&pred->next, item, TAG(STRIP_MARK(sitem->next), tag))) {
        haz_defer_free(sitem);
        return TRUE;
    }

insert中也可以更新pred->next的計數(shù)。

總結

無鎖的實現(xiàn),本質上都會依賴于CAS的互斥性。從頭實現(xiàn)一個lock free的數(shù)據(jù)結構,可以深刻感受到lock free實現(xiàn)的tricky。最終代碼可以從這里github獲取。代碼中為了簡單,實現(xiàn)了一個不是很強大的hazard pointer,可以參考之前的博文。

posted on 2015-05-05 19:47 Kevin Lynx 閱讀(22691) 評論(0)  編輯 收藏 引用 所屬分類: c/c++

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            一区二区欧美日韩| 亚洲视频大全| 欧美国产综合视频| 亚洲精品小视频| 亚洲欧美日韩精品在线| 国产精品日韩久久久久| 欧美亚洲在线视频| 免费h精品视频在线播放| 日韩视频免费大全中文字幕| 欧美视频一区在线观看| 性欧美8khd高清极品| 猛干欧美女孩| 亚洲午夜激情网站| 国产中文一区| 欧美精品一区三区| 亚洲欧美在线网| 亚洲国产成人av好男人在线观看| 亚洲美女在线国产| 国产日产欧美a一级在线| 久久亚洲色图| 亚洲一区精品视频| 欧美成人一区二区| 亚洲免费一在线| 亚洲国产成人精品久久| 欧美婷婷久久| 免费91麻豆精品国产自产在线观看| 一本色道久久88亚洲综合88| 六十路精品视频| 亚洲综合欧美| 99re6这里只有精品| 国产一区再线| 国产精品久久777777毛茸茸| 免费毛片一区二区三区久久久| 亚洲视频免费在线观看| 亚洲国产毛片完整版| 久久精品日韩一区二区三区| 在线亚洲电影| 91久久一区二区| 韩国av一区二区| 欧美无砖砖区免费| 奶水喷射视频一区| 久久精品一区| 亚洲欧美日韩成人| 99成人免费视频| 亚洲国产视频一区| 麻豆精品视频在线观看| 香蕉国产精品偷在线观看不卡| 亚洲精品一区二区在线观看| 狠狠综合久久av一区二区小说 | 加勒比av一区二区| 国产精品毛片大码女人| 欧美人体xx| 欧美激情亚洲| 猛干欧美女孩| 猫咪成人在线观看| 久久中文字幕一区| 久久精品国产免费看久久精品 | 久久精品导航| 欧美在线高清| 欧美亚洲一区二区三区| 午夜精品成人在线| 午夜免费久久久久| 亚洲女性喷水在线观看一区| 亚洲午夜在线观看视频在线| 99视频国产精品免费观看| 亚洲理论在线| 99精品视频一区| 一本一道久久综合狠狠老精东影业 | 亚洲免费中文字幕| 亚洲伊人观看| 午夜日本精品| 久久国产福利| 久久久久久久久蜜桃| 久久久亚洲精品一区二区三区 | 99视频一区二区| 9国产精品视频| 中文网丁香综合网| 亚洲自拍偷拍网址| 欧美一级欧美一级在线播放| 欧美在线观看日本一区| 久久久亚洲高清| 欧美96在线丨欧| 欧美日韩国产成人在线观看| 欧美日韩亚洲综合在线| 国产精品国产成人国产三级| 国产精品视频久久| 国产一区二区按摩在线观看| 永久免费毛片在线播放不卡| 亚洲人成网站在线播| 在线亚洲一区| 久久激情五月婷婷| 欧美成人首页| 亚洲另类在线一区| 亚洲欧美综合国产精品一区| 久久精品视频va| 欧美国产高潮xxxx1819| 欧美系列亚洲系列| 国产一区二区久久久| 亚洲国产综合在线| 亚洲综合日韩| 老司机午夜精品视频在线观看| 亚洲国产精彩中文乱码av在线播放| 亚洲日本欧美在线| 亚洲欧美中文另类| 欧美成人中文字幕| 国产久一道中文一区| 亚洲国产精品www| 亚洲一区精彩视频| 免费成人av在线| 亚洲天堂男人| 美女任你摸久久| 国产精品网站在线| 亚洲精品欧美在线| 国产伦精品一区二区| 另类酷文…触手系列精品集v1小说| 麻豆成人综合网| 亚洲品质自拍| 亚洲激情午夜| 精品盗摄一区二区三区| 国产精品一区二区久激情瑜伽| 亚洲天堂网在线观看| 欧美亚洲一级| 尤物精品在线| 国产精品午夜av在线| 欧美日韩色婷婷| 久久国产精品网站| 欧美亚洲视频一区二区| 亚洲自拍都市欧美小说| 亚洲精品男同| 99在线视频精品| 亚洲淫片在线视频| 亚洲已满18点击进入久久| 亚洲视频在线播放| 欧美一区成人| 久久久久久自在自线| 老司机aⅴ在线精品导航| 久久一二三国产| 欧美日韩国产区一| 国产亚洲毛片在线| 亚洲一级二级在线| 亚洲一区二区三区久久| 久久国产精品99久久久久久老狼 | 久久动漫亚洲| 欧美黄污视频| 国产伦精品一区二区三区视频黑人 | 麻豆freexxxx性91精品| 亚洲欧洲精品一区二区三区波多野1战4 | 欧美成人午夜激情| 一区二区成人精品| 久久电影一区| 亚洲深夜福利视频| 羞羞色国产精品| 欧美激情一二三区| 一区二区在线观看av| 亚洲男女自偷自拍| 亚洲电影第三页| 久久精品理论片| 国产视频综合在线| 先锋影音久久久| 亚洲每日在线| 欧美日韩中文字幕在线视频| 在线欧美日韩| 精品成人一区二区| 在线观看不卡| 久久精品视频免费| 久久精品欧美日韩| 亚洲高清色综合| 亚洲激情国产| 欧美人与性动交α欧美精品济南到| 伊人成人在线| 欧美一区在线直播| 一区二区三区免费看| 欧美色图一区二区三区| 国产精品99久久不卡二区| 日韩亚洲欧美一区二区三区| 亚洲男人的天堂在线观看| 在线视频精品| 国产精品一区二区黑丝| 久久国产精品99久久久久久老狼 | 国产精品草草| 午夜伦理片一区| 久久精品女人| 在线欧美视频| 日韩视频免费观看高清在线视频| 欧美三级不卡| 一区二区日韩| 在线中文字幕不卡| 亚洲麻豆av| 国外视频精品毛片| 亚洲美女性视频| 国产精品资源| 欧美激情视频在线免费观看 欧美视频免费一 | 欧美黄污视频| 欧美激情精品久久久六区热门| 伊人久久婷婷色综合98网| 欧美电影免费| 永久91嫩草亚洲精品人人| 亚洲午夜激情| 欧美一区二区三区日韩视频| 国产精品jizz在线观看美国|