深入浅出Redis 跳表的实现原理(redis的跳表实现原理)

深入浅出:Redis 跳表的实现原理

Redis 是一款高性能的开源内存数据存储系统。在 Redis 中,跳表(Skip List)被广泛应用于有序集合的实现中。今天,我们将深入浅出地了解 Redis 跳表的实现原理,帮助更好地理解 Redis 底层数据结构。

跳表的定义

跳表是一种可以支持快速查找、插入、删除的有序数据结构。跳表由于具有高效的查找性能、容易实现等优点,成为了 Redis 内部实现有序集合的一种常用数据结构。

跳表的结构

跳表的整体结构是一个多层的链表,每一层都是一个有序序列,每个节点除了记录本节点的值和指向下一个节点的指针外,还会记录该节点在高层链表中的指针。以下是一个三层跳表的示意图:

![skip_list](https://user-images.githubusercontent.com/62681965/133344905-0f814a59-86d2-4422-bc8b-23c77edc4213.png)

跳表的节点定义如下:

“`c

typedef struct skipListNode {

int score;

struct skipListNode *forward[1];

}skipListNode;


参数说明:

score:节点的分值,用于排序;
forward:前向指针数组,在每一层都指向下一个节点,共计 n 层,数组大小为 forward[0] 至 forward[n];
跳表查询

当跳表中存在 n 个节点时,对其中分值为 x 的节点进行查询,遍历的次数不超过 logn 代价最低。 执行跳表查询如下:

1.初始化指针
  从最高层开始,将指针指向对应节点。
2.开始查找
  从最高层开始,如果下一个节点的分值比目标分值小,就继续遍历下一个节点,直到找到查询的分值或者下一个节点的分值比目标分值大。
3.下移指针
  如果下一个节点的分值比目标分值大,就将指针下移一层,并重新执行步骤2。
跳表插入

在跳表中插入一个新节点时,需要找到插入位置并插入节点。插入节点有 50% 的概率会在某一层被插入,而在下一级序列不出现。执行跳表插入的步骤如下:

1. 通过查询找到待插入节点的位置,并预先设置层数。
  在插入节点之前,先从头开始查找要插入的位置,预先确定插入节点的所在层数。
2. 更新层信息
  插入新节点后,须更新跳表每层中指向相邻节点的指针,同时确定插入节点的层数。
3. 完成插入操作

跳表删除

在跳表已知的删除节点后,需要找到删除节点的前后节点。为了保持跳表整体的有序性,必须依次删除该节点在各个层中的指针,并将前后节点的指针互相连接。执行跳表删除的步骤如下:

1. 从最高层开始遍历找到要被删除的节点,并记录其所有前继节点。
2. 遍历所有前继节点,并更新掉该节点在上一层的指针,如下图所示,删除节点 16 后需要重建节点 14 在上一层的指针,直至跳表的最低层。
3. 删除指定的节点。

跳表的实现

以下是一个基于C语言实现的跳表操作的代码:

```c
#include
#include
#include
// 跳表结构体
typedef struct skipListNode {
int score;
struct skipListNode *forward[1];
}skipListNode;

// 跳表结构体
typedef struct skipList {
struct skipListNode *head;
struct skipListNode *tl;
int level; // 当前跳表最高级别
}skipList;
// 创建新的节点
skipListNode* newNode(int level, int score) {
skipListNode *node = (skipListNode*) malloc(sizeof(skipListNode) + level*sizeof(skipListNode*));
node->score = score;
return node;
}
// 创建新的跳表
skipList* createSkipList() {
int i;
skipList *list = (skipList*) malloc(sizeof(skipList));
skipListNode *head = newNode(32, 0);
list->head = head;
for(i = 0; i
head->forward[i] = NULL;
}
list->tl = NULL;
list->level = 1;
srand(time(0));
return list;
}
// 插入节点
void insert(skipList *list, int score) {
skipListNode *update[32];
skipListNode *current;
int i, level;
current = list->head;
for(i = list->level-1; i >= 0; i--) {
while(current->forward[i] && current->forward[i]->score
current = current->forward[i];
}
update[i] = current;
}
if(current->forward[0] && current->forward[0]->score == score) {
return;
}
level = rand()%32 + 1;
if(level > list->level ) {
for(i=list->level; i
update[i] = list->head;
}
list->level = level;
}
current = newNode(level, score);
for(i = 0; i
current->forward[i] = update[i]->forward[i];
update[i]->forward[i] = current;
}
}
// 删除节点
void delete(skipList *list, int score) {
skipListNode *update[32];
skipListNode *current;
int i;
current = list->head;
for(i = list->level-1; i >= 0; i--) {
while(current->forward[i] && current->forward[i]->score
current = current->forward[i];
}
update[i] = current;
}
current = current->forward[0];
if(current && current->score == score) {
for(i = 0; i level; i++) {
if(update[i]->forward[i] == current) {
update[i]->forward[i] = current->forward[i];
}
}
free(current);
while(list->level > 1 && list->head->forward[list->level-1] == NULL) {
list->level--;
}
}
}

// 查找节点
skipListNode* find(skipList *list, int score) {
skipListNode *current;
int i;
current = list->head;
for(i = list->level-1; i >= 0; i--) {
while(current->forward[i] && current->forward[i]->score
current = current->forward[i];
}
}
if(current->forward[0] && current->forward[0]->score == score) {
return current->forward[0];
} else {
return NULL;
}
}

在以上代码中,我们使用链表实现了跳表的添加节点、删除节点和查询节点的功能。

结语

跳表是一种高效的数据结构,其底层实现被广泛应用于 Redis 内部有序集合的实现中。跳表通过增加多个等级来实现高效的查找、插入和删除。希望今天的文章能够让你更好地理解 Redis 跳表的实现原理。

香港服务器首选树叶云,2H2G首月10元开通。
树叶云(www.IDC.Net)提供简单好用,价格厚道的香港/美国云服务器和独立服务器。IDC+ISP+ICP资质。ARIN和APNIC会员。成熟技术团队15年行业经验。

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/297934.html<

(0)
运维的头像运维
上一篇2025-05-22 17:06
下一篇 2025-05-22 17:07

相关推荐

  • virmach是什么?virmach主机评测及购买指南

    2026 年 Virmach 依然是高性价比 VPS 的首选,其核心优势在于提供基于 NVMe 的入门级方案与稳定的 KVM 架构,特别适合预算敏感型开发者及中小型企业,但需警惕其部分节点在跨境访问时的网络延迟波动,在 2026 年的云计算市场中,VPS 服务已从单纯的资源售卖转向“算力 + 网络 + 稳定性……

    2026-05-02
    0
  • BuyVMVPS测评,实测体验怎么样,BuyVMVPS测评推荐

    BuyVMVPS 在 2026 年依然是性价比极高的入门级选择,特别适合预算有限但追求高 I/O 性能的开发者与小型企业,但在高并发场景下需警惕其单线带宽限制,BuyVMVPS 核心架构与 2026 年市场定位在 2026 年的云计算市场中,BuyVMVPS 凭借独特的“按年付费”模式与高规格硬件配置,成功占据……

    2026-05-02
    0
  • hostsolutionsVPS测评,抗投诉实测表现,hostsolutionsVPS抗投诉能力如何?

    hostsolutions VPS 在 2026 年的抗投诉实测中表现卓越,其独特的“先斩后奏”容错机制配合高防节点,使其成为处理敏感业务(如海外营销、跨境支付)时优于传统廉价 VPS 的首选方案,但需注意其价格略高于行业平均水平,核心性能与抗投诉机制深度解析在 2026 年复杂的网络监管环境下,VPS 服务商……

    2026-05-02
    0
  • BuyVM服务器测评,实测体验好吗,BuyVM服务器怎么样

    BuyVM 服务器在 2026 年依然具备极高的性价比,特别适合需要低成本海外节点、搭建轻量级应用或进行跨境网络测试的用户,但其在高端游戏场景下的延迟表现略逊于顶级独服,在 2026 年的云主机市场,BuyVM 凭借“按需付费”与“固定 IP”的混合模式,依然占据着中小开发者与个人站长的核心生态位,对于寻找Bu……

    2026-05-02
    0
  • AlphaNine美国怎么样?AlphaNine美国官网入口

    AlphaNine 美国作为 2026 年高性能计算与边缘 AI 领域的领军品牌,其核心优势在于通过自研异构计算架构实现了 40% 的能效比提升,是解决企业级数据延迟与算力瓶颈的首选方案,尤其在 2026 年中美科技博弈背景下,其合规性与本土化服务能力成为关键决策点,AlphaNine 美国技术架构与 2026……

    2026-05-02
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注