無鎖(lock-free)數據結構
轉載自:http://www.limodev.cn/blog/archives/494
系統程序員成長計劃-并發(五)
提到并行計算通常都會想到加鎖,事實卻并非如此,大多數并發是不需要加鎖的。比如在不同電腦上運行的代碼編輯器,兩者并發運行不需要加鎖。在一臺電腦上同時運行的媒體播放放器和代碼編輯器,兩者并發運行不需要加鎖(當然系統調用和進程調度是要加鎖的)。在同一個進程中運行多個線程,如果各自處理獨立的事情也不需要加鎖(當然系統調用、進程調度和內存分配是要加鎖的)。在以上這些情況里,各個并發實體之間沒有共享數據,所以雖然并發運行但不需要加鎖。多線程并發運行時,雖然有共享數據,如果所有線程只是讀取共享數據而不修改它,也是不用加鎖的,比如代碼段就是共享的“數據”,每個線程都會讀取,但是不用加鎖。排除所有這些情況,多線程之間有共享數據,有的線程要修改這些共享數據,有的線程要讀取這些共享數據,這才是程序員需要關注的情況,也是本節我們討論的范圍。
在并發的環境里,加鎖可以保護共享的數據,但是加鎖也會存在一些問題:
-
由于臨界區無法并發運行,進入臨界區就需要等待,加鎖帶來效率的降低。
-
在復雜的情況下,很容易造成死鎖,并發實體之間無止境的互相等待。
-
在中斷/信號處理函數中不能加鎖,給并發處理帶來困難。
-
優先級倒置造成實時系統不能正常工作。低級優先進程拿到高優先級進程需要的鎖,結果是高/低優先級的進程都無法運行,中等優先級的進程可能在狂跑。
由于并發與加鎖(互斥)的矛盾關系,無鎖數據結構自然成為程序員關注的焦點,這也是本節要介紹的:
CPU提供的原子操作
大約在七八年前,我們用apache的xerces來解析XML文件,奇怪的是多線程反而比單線程慢。他們找了很久也沒有找出原因,只是證實使用多進程代替多線程會快一個數量級,在Windows上他們就使用了多進程的方式。后來移植到linux時候,我發現xerces每創建一個結點都會去更新一些全局的統計信息,比如把結點的總數加一,它使用的pthread_mutex實現互斥。這就是問題所在:一個XML文檔有數以萬計的結點,以50個線程并發為例,每個線程解析一個XML文檔,總共要進行上百萬次的加鎖/解鎖,幾乎所有線程都在等待,你說能快得了嗎?
當時我知道Windows下有InterlockedIncrement之類的函數,它們利用CPU一些特殊指令,保證對整數的基本操作是原子的。查找了一些資源發現Linux下也有類似的函數,后來我把所有加鎖去掉,換成這些原子操作,速度比多進程運行還快了幾倍。下面我們看++和—的原子操作在IA架構上的實現:
























單入單出的循環隊列。單入單出的循環隊列是一種特殊情況,雖然特殊但是很實用,重要的是它不需要加鎖。這里的單入是指只有一個線程向隊列里追加數據(push),單出只是指只有一個線程從隊列里取數據(pop),循環隊列與普通隊列相比,不同之處在于它的最大數據儲存量是事先固定好的,不能動態增長。盡管有這些限制它的應用還是相當廣泛的。這我們介紹一下它的實現:
數據下定義如下:










r_cursor指向隊列頭,用于取數據(pop)。w_cursor指向隊列尾,用于追加數據(push)。length表示隊列的最大數據儲存量,data表示存放的數據,[0]在這里表示變長的緩沖區(前面我們已經講過)。
創建函數






















這里我們要求隊列的長度大于1而不是大于0,為什么呢?排除長度為1的隊列沒有什么意義的原因外,更重要的原因是隊列頭與隊列尾重疊 (r_cursor= =w_cursor) 時,到底表示是滿隊列還是空隊列?這個要搞清楚才行,上次一個同事犯了這個錯誤,讓我們查了很久。這里我們認為隊列頭與隊列尾重疊時表示隊列為空,這與隊列初始狀態一致,后面在寫的時候始終保留一個空位,避免隊列頭與隊列尾重疊,這樣可以消除歧義了。
追加數據(push)























隊列頭和隊列尾之間還有一個以上的空位時就追加數據,否則返回失敗。
取數據(pop)




















隊列頭和隊列尾不重疊表示隊列不為空,取數據并移動隊列頭。
單寫多讀的無鎖數據結構
單寫表示只有一個線程去修改共享數據結構,多讀表示有多個線程去讀取共享數據結構。前面介紹的讀寫鎖可以有效的解決這個問題,但更高效的辦法是使用無鎖數據結構。思路如下:
就像為了避免顯示閃爍而使用的雙緩沖一樣,我們使用兩份數據結構,一份數據結構用于讀取,所有線程都可以在不加鎖的情況下讀取這個數據結構。另外一份數據結構用于修改,由于只有一個線程會修改它,所以也不用加鎖。
在修改之后,我們再交換讀/寫的兩個函數結構,把另外一份也修改過來,這樣兩個數據結構就一致了。在交換時要保證沒有線程在讀取,所以我們還需要一個讀線程的引用計數。現在我們看看怎么把前面寫的雙向鏈表改為單寫多讀的無鎖數據結構。
為了保證交換是原子的,我們需要一個新的原子操作CAS(compare and swap)。










數據結構







兩個鏈表,一個用于讀一個用于寫。rd_index_and_ref的最高字節記錄用于讀取的雙向鏈表的索引,低24位用于記錄讀取線程的引用記數,最大支持16777216個線程同時讀取,應該是足夠了,所以后面不考慮它的溢出。
讀取操作














修改操作





























先修改用于修改的雙向鏈表,修改完成之后等到沒有線程讀取時,交換讀/寫兩個鏈表,再修改另一個鏈表,此時兩個鏈表狀態保持一致。
稍做改進,對修改的操作進行加鎖,就可以支持多讀多寫的數據結構,讀是無鎖的,寫是加鎖的。
真正的無鎖數據結構
Andrei Alexandrescu的《Lock-FreeDataStructures》估計是這方面最經典的論文了,對他的方法我開始感到驚奇后來感到失望,驚奇的是算法的巧妙,失望的是無鎖的限制和代價。作者最后說這種數據結構只適用于WRRMBNTM(Write-Rarely-Read-Many -But-Not-Too-Many)的情況。而且每次修改都要拷貝整個數據結構(甚至多次),所以不要指望這種方法能帶來多少性能上的提高,唯一的好處是能避免加鎖帶來的部分副作用。有興趣的朋友可以看下這篇論文,這里我就不重復了。