Redis跳躍表

本文摘抄自redis 源碼剖析

跳躍表是一種隨機化的數(shù)據(jù),跳躍表以有序的方式在層次化的鏈表中保存元素,效率和平衡樹媲美 ——查找啦桌、刪除准颓、添加等操作都可以在對數(shù)期望時間下完成,并且比起平衡樹來說凉馆,跳躍表的實現(xiàn)要簡單直觀得多薪寓。
以下是個典型的跳躍表例子:



從圖中可以看到,跳躍表主要由以下部分構(gòu)成:

  • 表頭(head):負責維護跳躍表的節(jié)點指針澜共。
  • 跳躍表節(jié)點:保存著元素值向叉,以及多個層。
  • 層:保存著指向其他元素的指針嗦董。高層的指針越過的元素數(shù)量大于等于低層的指針母谎,為了提高查找的效率,程序總是從高層先開始訪問京革,然后隨著元素值范圍的縮小奇唤,慢慢降低層次幸斥。
  • 表尾:全部由NULL組成,表示跳躍表的末尾咬扇。

跳躍表的實現(xiàn)

為了滿足自身的功能需要甲葬,Redis對跳躍表進行了如下修改:

  • 允許重復的score
  • 值:多個不同的member的score 值可以相同
    進行對比操作時,不僅要檢查score值懈贺,還要檢查member:當score
    值可以重復時经窖,單靠score值無法判斷一個元素的身份,所以需要連member域都一并檢查才行梭灿。
    每個節(jié)點都帶有一個高度為1層的后退指針画侣,用于從表尾方向向表頭方向迭代:當執(zhí)行ZREVRANGE或ZREVRANGEBYSCORE這類以逆序處理有序集的命令時,就會用到這個屬性堡妒。

這個修改版的跳躍表由redis.h/zskiplist結(jié)構(gòu)定義:

typedef struct zskiplist {
     // 頭節(jié)點配乱,尾節(jié)點 
    struct zskiplistNode *header, *tail;
   // 節(jié)點數(shù)量 
   unsigned long length;
   // 目前表內(nèi)節(jié)點的最大層數(shù) int level;
} zskiplist;

跳躍表的節(jié)點由redis.h/zskiplistNode定義:

typedef struct zskiplistNode { 
    //member 對象
    robj *obj;
   // 分值
   double score;
  // 后退指針
    struct zskiplistNode *backward;
  // 層 
  struct zskiplistLevel {
   // 前進指針
   struct zskiplistNode *forward;
  // 這個層跨越的節(jié)點數(shù)量
   unsigned int span; 
  } level[];
} zskiplistNode;

跳躍表的應用

和字典、鏈表或者字符串這幾種在 Redis 中大量使用的數(shù)據(jù)結(jié)構(gòu)不同皮迟,跳躍表在 Redis 的唯一作用宪卿,就是實現(xiàn)有序集數(shù)據(jù)類型

跳躍表將指向有序集的score值和member
域的指針作為元素万栅,并以score 值為索引佑钾,對有序集元素進行排序。
舉個例子烦粒,以下代碼創(chuàng)建了一個帶有 3 個元素的有序集:

redis> ZADD s 6 x 10 y 15 z
(integer) 3
redis> ZRANGE s 0 -1 WITHSCORES
1) "x"
2) "6"
3) "y"
4) "10"
5) "z"
6) "15"

在底層實現(xiàn)中休溶,Redis 為x、y和z三個member分別創(chuàng)建了三個字符串扰她,值分別為double類型的6兽掰、10和15,然后用跳躍表將這些指針有序地保存起來徒役,形成這樣一個跳躍表:



為了方便展示孽尽,在圖片中我們直接將member和score值包含在表節(jié)點中,但是在實際的定義中忧勿,因為跳躍表要和另一個實現(xiàn)有序集的結(jié)構(gòu)(字典)分享member和score值杉女,所以跳躍表只保存指向member
和score的指針。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末鸳吸,一起剝皮案震驚了整個濱河市熏挎,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌晌砾,老刑警劉巖坎拐,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡哼勇,警方通過查閱死者的電腦和手機都伪,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來积担,“玉大人院溺,你說我怎么就攤上這事“跚幔” “怎么了?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵逐虚,是天一觀的道長聋溜。 經(jīng)常有香客問我,道長叭爱,這世上最難降的妖魔是什么撮躁? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮买雾,結(jié)果婚禮上把曼,老公的妹妹穿的比我還像新娘。我一直安慰自己漓穿,他們只是感情好嗤军,可當我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著晃危,像睡著了一般叙赚。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上僚饭,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天震叮,我揣著相機與錄音,去河邊找鬼鳍鸵。 笑死苇瓣,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的偿乖。 我是一名探鬼主播击罪,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼贪薪!你這毒婦竟也來了外邓?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤古掏,失蹤者是張志新(化名)和其女友劉穎损话,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡丧枪,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年光涂,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拧烦。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡忘闻,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出恋博,到底是詐尸還是另有隱情齐佳,我是刑警寧澤,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布债沮,位于F島的核電站炼吴,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏疫衩。R本人自食惡果不足惜硅蹦,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望闷煤。 院中可真熱鬧童芹,春花似錦、人聲如沸鲤拿。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽近顷。三九已至嗜价,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間幕庐,已是汗流浹背久锥。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留异剥,地道東北人瑟由。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像冤寿,于是被迫代替她去往敵國和親歹苦。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,792評論 2 345

推薦閱讀更多精彩內(nèi)容