Redis 有序集合 Zset的实现原理是什么?

Redis 有序集合 Zset的实现原理是什么?

重要内容

Zset通过两种编码方式实现,根据数据规模动态切换(默认阈值:元素个数 ≤ 128 且 单个元素大小 ≤ 64字节 为小规模数据,只要其中有一个不满足条件,则会切换编码方式)

Ziplist(压缩列表)

适用于小规模数据

  • 结构特点:连续内存存储,每个元素由 value 和 score 成对组成,按 score 升序排列
  • 优势:内存紧凑,无指针开销,适合少量数据的快速顺序访问

Skiplist(跳表)+ Hash(哈希表)

当数据规模超过阈值时自动切换为此结构,兼顾高效的范围查询与单点查找

  • 跳表(Skiplist)

    • 多层链表结构,节点包含多个前向指针(forward)和跨度(span),支持平均 O(logN) 的插入、删除和范围查询

    • 节点层数通过概率算法(如幂次定律)随机生成,高层节点稀疏,加速跳跃式查找

  • 哈希表(Hash)

    • 哈希表结构,存储 value 到 score 的映射,实现 O(1) 的单点查询(如:ZSCORE 命令)

扩展内容

有序集合的插入、删除和查找操作

插入操作

ziplist 插入

  1. 查找位置:遍历找到分值合适的位置 O(N)
  2. 插入数据:将新元素插入对应位置,触发内存重分配和数据移动
  3. 编码转换 :若插入后元素数量或大小超过阈值,转为 skiplist + hash

skiplist + hash 插入

  1. 哈希操作:检查成员是否存在,若存在需先删除旧值 O(1)
  2. 跳表插入
    • 从最高层开始查找插入位置,逐层向下直到最底层 O(logN)
    • 随机生成层高(如抛硬币算法),创建新节点并更新指针
  3. 哈希更新:将 value 到 score 写入哈希表 O(1)
删除操作

ziplist 删除

  1. 查找元素:遍历找到目标成员 O(N)
  2. 数据删除:移除元素并填充空隙,触发内存移动
  3. 编码转换:删除后若数据量过小,可能转回 ziplist

skiplist + dict 删除

  1. 字典查找:通过哈希表快速获取成员分值 O(1)
  2. 跳表删除
    • 从最高层开始查找节点,逐层向下直到最底层 O(logN
    • 调整前后节点的指针,释放节点内存
  3. 字典删除:从哈希表中移除成员 O(1)
查询操作

单点查询(获取分值)

  • ziplist:遍历所有元素 O(N)
  • skiplist + hash:通过哈希表直接获取分值 O(1)

范围查询(如 ZRANGE)

  • ziplist:顺序遍历底层链表 O(M),M为返回元素数
  • skiplist:利用跳表快速定位起始节点,遍历底层链表 O(logN + M)

排名查询(如 ZRANK)

  • ziplist:遍历计数直到找到成员 O(N)
  • skiplist:跳表遍历时累积跨度(span),直接计算排名 O(logN)

性能总结

操作 ziplist skiplist + hash
插入/删除 O(N) O(logN)(跳表)(哈希表 的查询 O(1) 可以忽略)
单点查询分值 O(N) O(1)(哈希表)
范围查询 O(M) O(logN + M)(跳表底层链表)
内存占用 低(连续存储) 较高(指针和哈希表开销)

应用场景

  1. 排行榜 :利用ZRANGEZREVRANGE快速获取排名
  2. 延迟队列 :以时间戳为score,定时扫描到期任务
  3. 实时统计 :如热点文章阅读量排序,通过ZINCRBY更新分值
  4. 范围查询:支持按分值区间快速筛选数据(如成绩分布)
相关推荐
怀璧其罪5 小时前
安装 OpenSSL 1.1.1 的完整脚本适用于 Ubuntu 22.04 系统
linux·服务器·数据库
ta叫我小白6 小时前
Spring Boot集成Redis并设置密码后报错: NOAUTH Authentication required
spring boot·redis
酷酷的崽7986 小时前
如何在SQL中高效使用聚合函数、日期函数和字符串函数:实用技巧与案例解析
服务器·数据库·sql
只做开心事7 小时前
MySQL索引
数据库·mysql
大数据追光猿7 小时前
【大模型面试知识】基础问题分析&总结
数据库·人工智能·深度学习·语言模型·ai编程
可爱de艺艺7 小时前
gin中使用GORM操作mysql数据库
数据库·mysql·gin
宋发元8 小时前
Redis&MySQL 3种常用的缓存读写策略详解
redis·mysql·缓存
@小张要努力8 小时前
基于STC89C52的8255并行口拓展实验
数据库·单片机·嵌入式硬件·学习·mongodb·51单片机·proteus
Faith_xzc8 小时前
存算分离是否真的有必要?从架构之争到 Doris 实战解析
大数据·数据库·数据仓库·架构·开源
从不删库的DBA8 小时前
Mysql-经典实战案例(10):如何用PT-Archiver完成大表的自动归档
数据库·mysql