有序集合zset的内部数据结构分析

2024-05-03 15:58

本文主要是介绍有序集合zset的内部数据结构分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  • 有序集合zset在内部可以使用压缩列表ziplist或者跳跃表skiplist来实现。
压缩列表
  • 如果使用压缩列表ziplist来实现,则键和分值紧凑相邻保存在压缩列表中,同时分值小的排在列表前面,分值大的排在列表后面。使用压缩列表ziplist来保存需要满足以下两个条件,否则使用跳跃表skiplist:
    • 集合元素少于128个;
    • 集合每个元素的键和分值都少于64个字节;
  • 即在集合元素较少,元素类型较小时,使用压缩列表,这样由于数据元素较少,故虽然压缩列表效率较低,但是基本不会有多大影响,而且可以充分利用压缩列表的连续内存和紧凑的数据结构实现来节省内存。
  • 以上两个条件可以在配置文件中分别通过zset-max-ziplist-entries和zset-max-ziplist-value这两个参数来修改。
跳跃表
  • 源码定义如下:

    // 有序集合
    typedef struct zset {dict *dict;zskiplist *zsl;
    } zset;
    
  • 如果是使用跳跃表skiplist,则会结合一个哈希字典dict来实现:

    • 即使用哈希字典dict来保存键和分值的映射,实现O(1)复杂度获取某个键的分值;通过跳跃表skiplist来根据分值排序来排序该集合,从而实现按分值排序,其中跳跃表的每个节点都同时包含分值和键。
    • 在内存使用方面,哈希字典dict和跳跃表skiplist是通过指针来共享键和分值的,故不会存在两份数据,节省了内存。
    • 如果只使用哈希字典来实现,则获取给定键对应的分值复杂度为O(1),但是如果每次获取有序集合,则需要获取整个集合然后排序,需要O(NlogN)的时间复杂度和O(N)的空间复杂度。
    • 如果只使用跳跃表skiplist来实现,则每次获取某个键的分值都需要在跳跃表来查找,时间复杂度为O(logN)。

这篇关于有序集合zset的内部数据结构分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/957099

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

MyBatis Plus大数据量查询慢原因分析及解决

《MyBatisPlus大数据量查询慢原因分析及解决》大数据量查询慢常因全表扫描、分页不当、索引缺失、内存占用高及ORM开销,优化措施包括分页查询、流式读取、SQL优化、批处理、多数据源、结果集二次... 目录大数据量查询慢的常见原因优化方案高级方案配置调优监控与诊断总结大数据量查询慢的常见原因MyBAT

分析 Java Stream 的 peek使用实践与副作用处理方案

《分析JavaStream的peek使用实践与副作用处理方案》StreamAPI的peek操作是中间操作,用于观察元素但不终止流,其副作用风险包括线程安全、顺序混乱及性能问题,合理使用场景有限... 目录一、peek 操作的本质:有状态的中间操作二、副作用的定义与风险场景1. 并行流下的线程安全问题2. 顺

Java JUC并发集合详解之线程安全容器完全攻略

《JavaJUC并发集合详解之线程安全容器完全攻略》Java通过java.util.concurrent(JUC)包提供了一整套线程安全的并发容器,它们不仅是简单的同步包装,更是基于精妙并发算法构建... 目录一、为什么需要JUC并发集合?二、核心并发集合分类与详解三、选型指南:如何选择合适的并发容器?在多

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

python语言中的常用容器(集合)示例详解

《python语言中的常用容器(集合)示例详解》Python集合是一种无序且不重复的数据容器,它可以存储任意类型的对象,包括数字、字符串、元组等,下面:本文主要介绍python语言中常用容器(集合... 目录1.核心内置容器1. 列表2. 元组3. 集合4. 冻结集合5. 字典2.collections模块