【每日算法】Day 16-1:跳表(Skip List)——Redis有序集合的核心实现原理(C++手写实现)
解锁O(log n)高效查询的链表奇迹!今日深入解析跳表的数据结构设计与实现细节,从基础概念到Redis级优化策略,彻底掌握这一平衡树的优雅替代方案。
一、跳表核心思想
跳表(Skip List) 是一种基于多层有序链表的概率型数据结构,核心特性:
多层结构:包含L0(完整数据层)到Lh(顶层索引层)
快速搜索:利用高层索引实现二分查找式跳跃
动态平衡:通过随机层数维持高效查询性能
与平衡树的对比优势:
| 特性 | 跳表 | 红黑树 |
|---|---|---|
| 实现复杂度 | 简单(无需旋转操作) | 复杂(需维护平衡) |
| 范围查询效率 | O(log n) + O(m) | O(log n + m) |
| 并发性能 | 更易实现锁细粒度控制 | 全局重平衡影响并发 |
| 内存占用 | 额外指针空间(约2倍) | 平衡信息存储 |
二、跳表节点定义
struct SkipListNode {int val;vector<SkipListNode*> next; // 多层后继指针SkipListNode(int v, int level) : val(v), next(level, nullptr) {}
};
三、跳表完整实现(C++)
1. 基础结构
class SkipList {
private:const float P = 0.25; // 节点晋升概率int maxLevel = 16; // 最大层数限制int curLevel = 0; // 当前最高层SkipListNode* head; // 头节点(哑节点)// 随机生成节点层数int randomLevel() {int level = 1;while ((rand() % 100) < P*100 && level < maxLevel)level++;return level;}public:SkipList() {head = new SkipListNode(INT_MIN, maxLevel);}~SkipList() {// 层序遍历销毁所有节点(代码略)}
};
2. 搜索操作
bool search(int target) {SkipListNode* curr = head;for (int i = curLevel-1; i >= 0; --i) {while (curr->next[i] && curr->next[i]->val < target) {curr = curr->next[i];}}curr = curr->next[0];return curr && curr->val == target;
}
3. 插入操作
void add(int num) {vector<SkipListNode*> update(maxLevel, head);SkipListNode* curr = head;// 记录每层需要更新的节点for (int i = curLevel-1; i >= 0; --i) {while (curr->next[i] && curr->next[i]->val < num) {curr = curr->next[i];}update[i] = curr;}// 创建新节点int newLevel = randomLevel();if (newLevel > curLevel) {for (int i = curLevel; i < newLevel; ++i)update[i] = head;curLevel = newLevel;}SkipListNode* newNode = new SkipListNode(num, newLevel);for (int i = 0; i < newLevel; ++i) {newNode->next[i] = update[i]->next[i];update[i]->next[i] = newNode;}
}
4. 删除操作
bool erase(int num) {vector<SkipListNode*> update(maxLevel, nullptr);SkipListNode* curr = head;// 定位待删除节点for (int i = curLevel-1; i >= 0; --i) {while (curr->next[i] && curr->next[i]->val < num) {curr = curr->next[i];}update[i] = curr;}curr = curr->next[0];if (!curr || curr->val != num) return false;// 更新各层指针for (int i = 0; i < curLevel; ++i) {if (update[i]->next[i] != curr) break;update[i]->next[i] = curr->next[i];}// 更新当前最高层while (curLevel > 1 && head->next[curLevel-1] == nullptr)curLevel--;delete curr;return true;
}
四、Redis的跳表优化策略
1. 特殊设计要点
-
晋升概率P=1/4:平衡空间与时间效率
-
最大层数=32:足够支持2^64元素的理论需求
-
ZSKIPLIST_MAXLEVEL:动态调整最高层数
-
双向指针:支持反向遍历(Redis 5.0+)
2. 存储结构图示
Redis跳表节点结构:
+------------+-----------+-------+-------+-----+-------+
| 成员对象 | 分值(score) | backward | level[] | ... |
+------------+-----------+-------+-------+-----+-------+
五、大厂真题实战
真题1:设计排行榜系统(某大厂2024面试)
需求:
实时维护玩家分数排名,支持:
-
更新玩家分数
-
查询Top N玩家
-
查询玩家排名
跳表解法:
class Leaderboard {
private:struct Node {int playerId;int score;// 重载比较运算符bool operator<(const Node& other) const {return score > other.score; // 按分数降序}};SkipList<Node> skipList;unordered_map<int, SkipListNode<Node>*> cache;public:void addScore(int playerId, int score) {if (cache.count(playerId)) {auto node = cache[playerId];int oldScore = node->val.score;skipList.erase({playerId, oldScore});score += oldScore;}auto newNode = skipList.add({playerId, score});cache[playerId] = newNode;}vector<int> top(int K) {vector<int> res;auto curr = skipList.head->next[0];while (K-- && curr) {res.push_back(curr->val.playerId);curr = curr->next[0];}return res;}
};
真题2:时间序列数据库索引(某大厂2023笔试)
需求:
高效查询时间范围内的数据点
跳表变种设计:
-
将时间戳作为排序键
-
在高层索引中存储时间区间统计量(如最大值/最小值)
-
范围查询时利用高层索引快速定位起始点
六、复杂度与优化对比
| 操作 | 时间复杂度 | 空间复杂度 | 优化方向 |
|---|---|---|---|
| 插入 | 平均O(log n) | O(n) | 调整晋升概率P |
| 删除 | O(log n) | O(n) | 延迟删除优化 |
| 查询 | O(log n) | O(n) | 增加高层索引密度 |
| 范围查询 | O(log n + m) | O(n) | 双向指针优化 |
七、常见误区与调试技巧
-
层数分配不均:随机数生成器质量影响性能(建议使用MT19937)
-
指针未初始化:新节点next数组需全部置空
-
内存泄漏:需分层遍历释放所有节点
-
调试技巧:
-
可视化打印跳表结构
-
为节点添加唯一ID辅助调试
-
边界测试(空表/单节点/连续插入相同值)
-
进阶学习资源:
-
Redis源码src/t_zset.c中的zskiplist实现
-
《算法导论》跳表复杂度证明
-
Paper: "Skip Lists: A Probabilistic Alternative to Balanced Trees"
LeetCode真题训练:
-
1206. 设计跳表
-
632. 最小区间(多指针+跳表优化)
相关文章:
【每日算法】Day 16-1:跳表(Skip List)——Redis有序集合的核心实现原理(C++手写实现)
解锁O(log n)高效查询的链表奇迹!今日深入解析跳表的数据结构设计与实现细节,从基础概念到Redis级优化策略,彻底掌握这一平衡树的优雅替代方案。 一、跳表核心思想 跳表(Skip List) 是一种基于多层有序链表的概率型数…...
前沿科技:3D生成领域技术与应用分析
以下是关于3D生成领域的详细分析,涵盖技术发展、应用场景、挑战与未来趋势、市场动态及典型案例: 一、技术发展与核心方法 3D表示方法 显式表示:包括点云、网格(三角形或四边形)和分层深度图像(LDI),适合直接操作和渲染,但细节复杂度高。 隐式表示:如神经辐射场(NeR…...
Spring Boot 3.4.3 基于 JSqlParser 和 MyBatis 实现自定义数据权限
前言 在企业级应用中,数据权限控制是保证数据安全的重要环节。本文将详细介绍如何在 Spring Boot 3.4.3 项目中结合 JSqlParser 和 MyBatis 实现灵活的数据权限控制,通过动态 SQL 改写实现多租户、部门隔离等常见数据权限需求。 一、环境准备 确保开发环境满足以下要求: …...
GO语言学习(14)GO并发编程
目录 🌈前言 1.goroutine🌟 2.GMP模型🌟 2.1 GMP的由来☀️ 2.2 什么是GMP☀️ 3.channel 🌟 3.1 通道声明与数据传输💥 3.2 通道关闭 💥 3.3 通道遍历 💥 3.4 Select语句 Ǵ…...
【Audio开发二】Android原生音量曲线调整说明
一,客制化需求 客户方对于音量加减键从静音到最大音量十五个档位区域的音量变化趋势有定制化需求。 二,音量曲线调试流程 Android根据不同的音频流类型定义不同的曲线,曲线文件存放在/vendor/etc/audio_policy_volumes.xml或者default_volu…...
sass报错,忽略 Sass 弃用警告,降级版本
最有效的方法是创建一个 .sassrc.json 文件来配置 Sass 编译器。告诉 Sass 编译器忽略来自依赖项的警告消息。 解决方案: 1. 在项目根目录创建 .sassrc.json 文件: {"quietDeps": true }这个配置会让 Sass 编译器忽略所有来自依赖项&#x…...
spring-security原理与应用系列:HttpSecurity.filters
目录 AnyRequestMatcher WebSecurityConfig HttpSecurity AbstractInterceptUrlConfigurer AbstractAuthenticationProcessingFilter 类图 在前面的文章《spring-security原理与应用系列:securityFilterChainBuilders》中,我们遗留了一个问题&…...
JVM生产环境问题定位与解决实战(六):总结篇——问题定位思路与工具选择策略
本文已收录于《JVM生产环境问题定位与解决实战》专栏,完整系列见文末目录 引言 在前五篇文章中,我们深入探讨了JVM生产环境问题定位与解决的实战技巧,从基础的jps、jmap、jstat、jstack、jcmd等工具,到JConsole、VisualVM、MAT的…...
数据仓库项目启动与管理
数据仓库项目启动与管理 确定项目 评估项目就绪情况 项目就绪的三个条件 强力型高级业务管理发起人 对数据仓库解决方案的影响有先见之明是所在组织内有影响的领导者要求严格,但是又比较现实,会为其他成员提供强力支持 强制型业务动机 数据仓库系统和战略性业务动机紧密结合…...
并行治理机制对比:Polkadot、Ethereum 与 NEAR
治理是任何去中心化网络的基础。它塑造了社区如何发展、如何为创新提供资金、如何应对挑战以及如何随着时间的推移建立信任。随着 Web3 的不断发展,决定这些生态系统如何做出决策的治理模型也在不断发展。 在最近的一集的【The Decentralized Mic】中, Polkadot 汇…...
利用 PHP 爬虫按关键字搜索淘宝商品
在当今数字化时代,网络爬虫技术已成为获取网络数据的重要手段之一。淘宝作为国内最大的电商平台之一,拥有海量的商品信息。通过 PHP 爬虫技术,我们可以实现按关键字搜索并抓取淘宝商品信息。以下将详细介绍如何使用 PHP 实现这一功能。 一、…...
在未归一化的线性回归模型中,特征的尺度差异可能导致模型对特征重要性的误判
通过数学公式来更清晰地说明归一化对模型的影响,以及它如何改变特征的重要性评估。 1. 未归一化的情况 假设我们有一个线性回归模型: y β 0 β 1 x 1 β 2 x 2 ϵ y \beta_0 \beta_1 x_1 \beta_2 x_2 \epsilon yβ0β1x1β2x2ϵ 其…...
TDengine tar.gz和docker两种方式安装和卸载
下载地址 3.1.1.0 Linux版本 安装包 下载地址 3.1.1.0 docker 镜像 下载地址 3.1.1.0 Window客户端 1. 将文件上传至服务器后解压 tar -zxvf TDengine-server-3.1.1.0-Linux-x64.tar.gz 2. tar.gz安装 解压文件后,进入相应子目录,执行其中的 install.…...
【STM32设计】基于STM32的智能门禁管理系统(指纹+密码+刷卡+蜂鸣器报警)(代码+资料+论文)
本课题为基于单片机的智能门禁系统,整个系统由AS608指纹识别模块,矩阵键盘,STM32F103单片机,OLED液晶,RFID识别模块,继电器,蜂鸣器等构成,在使用时,用户可以录入新的指纹…...
贪心算法,其优缺点是什么?
什么是贪心算法? 贪心算法(Greedy Algorithm)是一种在每一步选择中都采取在当前状态下最优(局部最优)的选择,从而希望导致全局最优解的算法策略。 它不像动态规划那样考虑所有可能的子问题,而是做出局部最优选择,依赖这些选择来…...
java知识梳理(二)
一.lambda表达式 作用:Lambda 表达式在 Java 8 引入,主要用于简化匿名内部类的写法,特别是在函数式编程场景中,比如 函数式接口、流式 API(Streams)、并发编程等。它让 Java 代码更简洁、可读性更强&#x…...
鸿蒙Flutter实战:20. Flutter集成高德地图,同层渲染
本文以同层渲染为例,介绍如何集成高德地图 完整代码见 Flutter 鸿蒙版 Demo 概述 Dart 侧 核心代码如下,通过 OhosView 来承载原生视图 OhosView(viewType: com.shaohushuo.app/customView,onPlatformViewCreated: _onPlatformViewCreated,creation…...
c++中%符号使用的注意事项/易错点
在C中,% 是取模运算符(modulus operator),用于计算两个数相除后的余数。虽然它的用法看起来简单,但在实际编程中有一些需要注意的细节和易错点。以下是关键注意事项: 1. 操作数必须为整数类型 % 只能用于整…...
AI辅助下基于ArcGIS Pro的SWAT模型全流程高效建模实践与深度进阶应用
目前,流域水资源和水生态问题逐渐成为制约社会经济和环境可持续发展的重要因素。SWAT模型是一种基于物理机制的分布式流域水文与生态模拟模型,能够对流域的水循环过程、污染物迁移等过程进行精细模拟和量化分析。SWAT模型目前广泛应用于流域水文过程研究…...
Java 基础-30-单例设计模式:懒汉式与饿汉式
在软件开发中,单例设计模式(Singleton Design Pattern)是一种常用的设计模式,它确保一个类只有一个实例,并提供一个全局访问点。这种模式通常用于管理共享资源(如数据库连接池、线程池等)或需要…...
尚语翻译图册翻译|专业图册翻译|北京专业翻译公司推荐|专业文件翻译报价
内容概要 尚语翻译公司聚焦多语种产品图册翻译的竞价推广服务,通过行业垂直化运营构建差异化竞争力。其核心服务覆盖机械制造、医疗器械、电子元件三大领域,依托ISO 17100认证的翻译流程和Trados术语管理系统,实现技术文档的精准转化。为提升…...
杂篇-行业分类一二-2(通、专用设备制造,汽车制造)
接上篇, 本篇列举制造业中另外几个细分行业:通用设备制造,专用设备制造,汽车制造业。 一、通用设备制造 分类 序号 类别名称 说明 1 锅炉及原动设备制造 1 锅炉及辅助设备制造 指各种蒸汽锅炉、汽化锅炉,以及…...
[笔记.AI]大模型训练 与 向量值 的关系
(借助 DeepSeek-V3 辅助生成) 大模型在训练后是否会改变向量化的值,取决于模型的训练阶段和使用方式。以下是详细分析: 1. 预训练阶段:向量化值必然改变 动态调整过程: 在预训练阶段(如BERT、…...
LeetCode 解题思路 30(Hot 100)
解题思路: 递归参数: 生成括号的对数 n、结果集 result、当前路径 path、左括号数 open、右括号数 close。递归过程: 当当前路径 path 的长度等于 n * 2 时,说明已经生成有效括号,加入结果集。若左括号数小于 n&…...
Java EE(18)——网络原理——应用层HTTP协议
一.初识HTTP协议 HTTP(HyperText Transfer Protocol,超文本传输协议)是用于在客户端(如浏览器)和服务器之间传输超媒体文档(如HTML)的应用层协议。 HTTP协议发展至今发布了多个版本,其中1.0,1.…...
强大而易用的JSON在线处理工具
强大而易用的JSON在线处理工具:程序员的得力助手 在当今的软件开发世界中,JSON(JavaScript Object Notation)已经成为了数据交换的通用语言。无论是前端还是后端开发,我们都经常需要处理、验证和转换JSON数据。今天&a…...
Qt笔记----》不同环境程序打包
文章目录 概要1、windows环境下打包qt程序2、linux环境下打包qt程序2.1、程序目录2.2、创建一个空文件夹2.3、添加依赖脚本2.4、打包过程2.4.1、添加程序依赖库2.4.2、添加Qt相关依赖库 概要 qt不同运行环境下打包方式:windows/linux 1、windows环境下打包qt程序 …...
企业服务器备份软件,企业服务器备份的方法有哪些?
企业服务器备份需综合考虑数据量、业务连续性要求(RTO/RPO)、合规性及成本等因素。以下是分场景的工具和方法指南: 一、备份软件推荐 1. 80KM备份软件 80KM备份软件可以进行很复杂的备份方式,也可以内网对内网备份、还能内网的…...
Vue3 表单
Vue3 表单 随着前端技术的发展,Vue.js 作为一款流行的前端框架,不断更新迭代,以适应更高效、更便捷的开发需求。Vue3 作为 Vue.js 的第三个主要版本,引入了许多新特性和改进,其中包括对表单处理机制的优化。本文将深入探讨 Vue3 表单的使用方法、技巧以及注意事项。 1. …...
html5炫酷图片悬停效果实现详解
html5炫酷图片悬停效果实现详解 这里写目录标题 html5炫酷图片悬停效果实现详解项目介绍技术栈核心功能实现1. 页面布局2. 图片容器样式3. 炫酷悬停效果缩放效果倾斜效果模糊效果旋转效果 4. 悬停文字效果5. 性能优化6. 响应式设计 项目亮点总结 项目介绍 本文将详细介绍如何使…...
