《Java后端体系系列》之Redis跳跃表是如何跳的

原文发布在语雀

一、定义
  1. 跳跃表是一种有序的数据结构,它通过在每个节点中维持多个指向其它节点的指针来达到快速访问节点的目的。
    如下图:节点1有指向节点2、节点3、节点4的指针,这样如果要访问节点4的数据,节点1可以直接通过指向节点4的指针来快速查询节点4的数据,而不需要依次遍历所有的节点。
    在这里插入图片描述

  2. 跳跃表的时间复杂度平均位OlogN,最坏为O(N)。在大多数情况下跳跃表的效率可以和平衡树相媲美,并且跳跃表的实现比平衡树更加简单。
    下图中实现了两个数据结构,平衡树和跳跃表(跳跃表中的数据是顺序存储)
    如果要查询4这个数据,那么在平衡树中走过7--->2--->4,那么在跳跃表中走过节点1--->节点4,或者节点1--->节点2--->节点4
    在这里插入图片描述

  3. Redis使用跳跃表作为有序集合键的底层之一,如果一个有序集合包含元素数量比较多,或者有序集合中的元素是比较长的字符串,Redis就会使用跳跃表来作为有序集合键的底层实现。

  4. Redis在两个地方使用了跳跃表:一、有序集合键;二、集群节点中用作内部数据结构。

二、实现
2.1、结构

Redis跳跃表由redis.h/zskiplistNode和redis.h/zskiplist两个结构定义,其中zskiplistNode结构用于表示跳跃表节点,而zskiplist结构则用于保存跳跃表节点的相关信息:节点数量、表头节点和表尾节点的指针等。
结构图如下:下图中有一个zskiplist和四个zskiplistNode
在这里插入图片描述

  • zskiplist中包含四个属性分别为:
  1. header:指向跳跃表的表头节点
  2. tail:指向跳跃表的表尾节点
  3. level:记录跳跃表中层数最大的节点的层数
  4. length:记录跳跃表的长度(跳跃表中包含节点的数量去掉表头)
  • zskiplistNode包含四个属性分别为:
  1. 层(level):每个节点中用L1、L2标记的块被是每个节点的层,每层有两个属性:前进指针和跨度
    a. 前进指针:前进指针用于访问指向的其它节点
    b. 跨度:记录当前指针指向的节点和当前节点的距离(例如节点1中指向节点4的跨度为3)
  2. 后退(backward):每个节点中标出的BW就是节点的后退指针,BW指向当前节点的前一个节点
  3. 分值(score):节点中存储的值,例如节点1中存储的2就是节点1的分值
  4. 成员对象(obj):节点中的O1、O2、O3代表节点保存的成员对象
2.2、代码定义
typedef struct zskiplistNode{
//后退指针
struct zskiplistNode *backward;
//分值
double score;
//成员对象
robj *obj;
//层
struct zskiplistLevel{
//前进指针
struct zskiplistNode *forward;
//跨度
unsigned int span;
} level[];
} zskiplistNode;
三、总结
  • 跳跃表节点的level数组包含多个元素,每个元素都包含一个指向其它节点的指针,程序可以通过这些层来加快访问其它节点的速度,一般来说层的数量越多,访问其它节点的速度越快。
  • 指向NULL的所有前进指针跨度为0,因为它们没有连向其它任何节点。
  • 遍历操作只使用前进指针,跨度是用来计算排位的。
  • 每个节点只有一个后退指针,每次只能后退到前一个节点,不能跨越后退。
  • header和tail是跳跃表中的表头和表尾,所以表头和表尾的复杂度为O1. 跳跃表中的节点按照分值的大小进行排序的,分值相同时,节点会按照成员对象的大小进行排序(都是从小到大)。
  • 跳跃表中多个节点可以保存相同的分值,但是每个节点的成员对象必须是唯一的。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值