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

loop_in_codes

低調(diào)做技術(shù)__歡迎移步我的獨(dú)立博客 codemaro.com 微博 kevinlynx

無(wú)鎖有序鏈表的實(shí)現(xiàn)

無(wú)鎖有序鏈表可以保證元素的唯一性,使其可用于哈希表的桶,甚至直接作為一個(gè)效率不那么高的map。普通鏈表的無(wú)鎖實(shí)現(xiàn)相對(duì)簡(jiǎn)單點(diǎn),因?yàn)椴迦朐乜梢栽诒眍^插,而有序鏈表的插入則是任意位置。

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

主要問(wèn)題

鏈表的主要操作包含insertremove,先簡(jiǎn)單實(shí)現(xiàn)一個(gè)版本,就會(huì)看到問(wè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的時(shí)候,其可能被其他線程移除。甚至,在l_find過(guò)程中,其每一個(gè)元素都可能被移除。問(wèn)題在于,任何時(shí)候拿到一個(gè)元素時(shí),都不確定其是否還有效。元素的有效性包括其是否還在鏈表中,其指向的內(nèi)存是否還有效。

解決方案

通過(guò)為元素指針增加一個(gè)有效性標(biāo)志位,配合CAS操作的互斥性,就可以解決元素有效性判定問(wèn)題。

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

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

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可能被刪除,此時(shí)pred->next被標(biāo)記
            //    pred->next != item,該CAS會(huì)失敗,失敗后重試
            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前先標(biāo)記item->next,如果CAS失敗,那么情況同insert一樣,有其他線程在find之后
            //    刪除了item,失敗后重試
            if (!CAS(&item->next, inext, MARK(inext))) {
                continue;
            }
            // C. 對(duì)同一個(gè)元素item刪除時(shí),只會(huì)有一個(gè)線程成功走到這里
            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);
            /* 
             如果已被標(biāo)記,那么緊接著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之類(lèi)的函數(shù)是一個(gè)hazard pointer實(shí)現(xiàn),用于支持多線程下內(nèi)存的GC。上面的代碼中,要?jiǎng)h除一個(gè)元素item時(shí),會(huì)標(biāo)記item->next,從而使得insert時(shí)中那個(gè)CAS不需要做任何調(diào)整。總結(jié)下這里的線程競(jìng)爭(zhēng)情況:

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

ABA問(wèn)題

ABA問(wèn)題還是存在的,insert中:

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

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

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

// 最低位留作刪除標(biāo)志
    #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的實(shí)現(xiàn):

/* 先標(biāo)記再刪除 */
    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的計(jì)數(shù)。

總結(jié)

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

posted on 2015-05-05 19:47 Kevin Lynx 閱讀(22688) 評(píng)論(0)  編輯 收藏 引用 所屬分類(lèi): 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>
            欧美激情一区二区三区蜜桃视频 | 久久精品国产亚洲一区二区三区| 亚洲国产裸拍裸体视频在线观看乱了中文| 在线视频亚洲一区| 亚洲精品一区二区三区不| 久久久久久久精| 国产精品综合久久久| 亚洲天堂网在线观看| 亚洲夜晚福利在线观看| 欧美日韩亚洲视频| 艳妇臀荡乳欲伦亚洲一区| 一区二区三区国产在线| 欧美日韩国产成人在线| 最新日韩在线视频| 日韩视频一区二区三区在线播放免费观看 | 亚洲午夜伦理| 亚洲一区日韩| 国产精品免费网站| 欧美一区二区三区免费在线看| 欧美在线播放一区二区| 国产视频在线观看一区二区| 午夜影院日韩| 久久免费偷拍视频| 亚洲夫妻自拍| 欧美精品免费观看二区| 日韩视频一区二区三区在线播放| 99国产麻豆精品| 国产精品免费一区二区三区在线观看 | 久久久精品日韩| 欧美成人午夜77777| 亚洲精品日韩久久| 欧美午夜不卡影院在线观看完整版免费| 99这里有精品| 久久精品国产精品亚洲| 亚洲福利视频一区| 欧美日韩国产综合视频在线观看中文 | 久久精品中文字幕免费mv| 在线欧美电影| 欧美日韩国产三级| 亚洲淫片在线视频| 欧美福利电影在线观看| 亚洲视频网站在线观看| 国产亚洲毛片在线| 欧美大片一区二区三区| 亚洲欧美三级伦理| 亚洲国产成人精品久久久国产成人一区| 一本久久综合亚洲鲁鲁五月天| 国产精品爽黄69| 蜜桃久久av一区| 夜夜精品视频一区二区| 久久精品国产免费| 夜久久久久久| 国内精品久久久久久久影视蜜臀 | 久久亚洲一区二区三区四区| 亚洲黄网站黄| 久久精品国产91精品亚洲| 亚洲精品久久久蜜桃| 国产精品亚洲片夜色在线| 欧美成人有码| 久久精品99| 国产精品每日更新| 亚洲精品一区二区三区不| 欧美一区二区三区免费视| 亚洲国产毛片完整版 | 国产婷婷精品| 欧美极品一区二区三区| 欧美一级片久久久久久久| 亚洲高清视频在线| 久久蜜桃精品| 欧美在线观看网站| 国产精品99久久久久久白浆小说| 海角社区69精品视频| 欧美性一二三区| 欧美成人精品在线| 欧美在线网站| 亚洲一区二区三区高清| 亚洲精品美女免费| 欧美护士18xxxxhd| 久久亚洲国产精品一区二区| 亚洲欧美日本伦理| 亚洲精品中文字幕在线| 在线观看亚洲视频| 国产综合色产| 国产伦理一区| 国产精品专区一| 欧美日韩综合在线| 欧美精品精品一区| 欧美成人乱码一区二区三区| 久久久91精品国产一区二区三区| 午夜亚洲福利| 亚洲欧美视频在线观看| 在线视频欧美日韩| 99国产精品视频免费观看一公开| 最新日韩欧美| 亚洲精品欧美极品| 亚洲国产专区| 欧美国产激情二区三区| 免费精品视频| 欧美华人在线视频| 亚洲国产欧美一区二区三区久久| 裸体素人女欧美日韩| 久久精品99国产精品日本| 性欧美大战久久久久久久免费观看 | 亚洲日本欧美日韩高观看| 伊人狠狠色丁香综合尤物| 激情五月综合色婷婷一区二区| 国产一区二区三区电影在线观看 | 欧美色另类天堂2015| 欧美日韩综合视频网址| 欧美人与性动交a欧美精品| 欧美精品一区二区三区一线天视频 | 牛夜精品久久久久久久99黑人| 美日韩在线观看| 欧美不卡视频一区发布| 欧美二区在线看| 欧美日韩视频在线一区二区| 欧美久久久久| 欧美视频1区| 国产精品在线看| 影音先锋日韩精品| 亚洲日韩成人| 亚洲一区二区三区在线看| 亚洲欧美国产精品桃花| 久久aⅴ乱码一区二区三区| 最新中文字幕一区二区三区| 亚洲欧洲一区二区三区久久| 99re6热只有精品免费观看| 夜夜爽99久久国产综合精品女不卡| 亚洲桃色在线一区| 久久久www| 亚洲国产欧美一区二区三区同亚洲 | 欧美资源在线观看| 久久综合综合久久综合| 欧美成人免费全部| 国产精品豆花视频| 国外成人在线视频网站| 99伊人成综合| 久久riav二区三区| 欧美高潮视频| 亚洲一区精品电影| 免费观看成人鲁鲁鲁鲁鲁视频| 欧美日韩国产综合在线| 国产亚洲一二三区| 99亚洲一区二区| 久久久亚洲精品一区二区三区| 亚洲激情一区二区| 亚洲欧美中文日韩在线| 裸体女人亚洲精品一区| 欧美日韩在线精品| 影音先锋亚洲精品| 午夜精品区一区二区三| 欧美成人情趣视频| 亚洲欧美一区二区三区在线| 欧美韩日一区二区三区| 国产欧美精品日韩| 一区二区三区成人| 麻豆freexxxx性91精品| 亚洲网站啪啪| 欧美国产一区二区在线观看| 国产精品一区二区视频| 亚洲欧洲精品成人久久奇米网| 欧美在线国产| 在线成人h网| 亚洲欧美另类中文字幕| 亚洲国产日韩欧美在线图片 | 一区二区三区四区国产精品| 久久久久www| 国产精品久久精品日日| 亚洲美女视频网| 欧美波霸影院| 欧美专区日韩专区| 国产精品欧美日韩一区| 一本久久a久久精品亚洲| 免费在线日韩av| 欧美在线高清视频| 国产精品实拍| 亚洲一区二区三区中文字幕在线| 亚洲国产精品一区二区第一页| 久久久91精品国产一区二区三区 | 午夜精品久久久| 91久久久久久| 久久综合色播五月| 亚洲自拍偷拍麻豆| 欧美午夜激情视频| 一本在线高清不卡dvd| 欧美国产三级| 另类亚洲自拍| 亚洲国产一区二区三区青草影视| 久久午夜精品一区二区| 小处雏高清一区二区三区 | 免播放器亚洲一区| 国内综合精品午夜久久资源| 久久精品女人| 欧美在线影院在线视频| 国产精品素人视频| 欧美一区二区三区视频在线| 亚洲曰本av电影| 国产欧美va欧美va香蕉在| 黄色综合网站| 噜噜噜91成人网|