平衡树:为什么Redis内部实现用跳跃表
摘要:Redis使用跳跃表(skiplist)作为有序集合(zset)的底层实现之一。
本文分享自华为云社区《5分钟了解Redis的内部实现跳跃表(skiplist)》,作者:万猫学社。
跳跃表简介
跳跃表(skiplist)是一个有序的数据结构,它通过在每个节点维护不同层次指向后续节点的指针,以达到快速访问指定节点的目的。跳跃表在查找指定节点时,平均时间复杂度为,最坏时间复杂度为O(N)。
Redis使用跳跃表(skiplist)作为有序集合(zset)的底层实现之一。当有序集合的元素个数大于等于zset-max-ziplist-entries(默认为128个),或者每个元素成员的长度大于等于zset-max-ziplist-value(默认为64字节)的时候,使用跳跃表和哈希表作为有序集合的内部实现。
举个例子,我们使用zadd命令创建一个以跳跃表为实现的有序集合:
127.0.0.1:6379> zadd one-more-zset 1 long-long-long-long-long-long-long-long-long-long-long-long-long-long
(integer) 1
127.0.0.1:6379> zrange one-more-zset 0 -1
1) "long-long-long-long-long-long-long-long-long-long-long-long-long-long"
127.0.0.1:6379> object encoding one-more-zset
"skiplist"
跳跃表的实现
在Redis中的跳跃表是由zskiplist结构表示的,zskiplist结构包含由多个跳跃表节点组成的双向链表,每一个跳跃表节点都保存着元素成员和对应的分钟。下面我们一个一个地详细了解一下。
zskiplist结构
跳跃表是由zskiplist结构表示的,它包含以下几个属性:
- header属性: 指向头部跳跃表节点的指针。
- tail属性:指向尾部跳跃表节点的指针。
- level属性:表示跳跃表中层数最大的节点的层数,表头节点的层数不计算在内。
- length属性:表示跳跃表中的节点总数。
跳跃表节点的结构
跳跃表节点使用zskiplistNode结构表示,它包含以下几个属性:
- level属性:表示层的数组,数组中每个项使用zskiplistLevel结构表示,它包含以下两个属性:
- forward属性:指向位于表尾方向其他节点的指针。
- span属性:当前节点到forward指向的节点跨越了多少个节点。
- backward属性:指向当前节点的前一个节点的指针。
- obj属性:指向元素成员的指针。
- score属性:当前元素成员对应的分数。
图解跳跃表
说了这么多,都比较抽象不容易理解,我们来举个例子:
这就是一个跳跃表的内部结构,其中有4个元素,键分别是:万、猫、学、社。
为什么不使用平衡树?
跳跃表以有序的方式在层次化的链表中保存元素, 在大多数情况下,跳跃表的效率可以和平衡树媲美,查找、删除、添加等操作都可以在对数期望时间下完成, 并且比起平衡树来说, 跳跃表的实现要简单直观得多。所以在Redis中没有使用平衡树,而是使用了跳跃表。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
-
上一篇
TypeScript里string和String,真不是仅仅是大小写的区别
摘要:通常来说,string表示原生类型,而String表示对象。 本文分享自华为云社区《TypeScript里string和String的区别》,作者:gentle_zhou 。 背景 与JavaScript语言不同的是,TypeScript使用的是静态类型,比如说它指定了变量可以保存的数据类型。如下图所示,如果在JS中,指定变量可以保存的数据类型,会报错:“类型注释只可以在TS文件中被使用”: TypeScript是JavaScript的超集(superset),TypeScript需要编译(语法转换)生成JavaScript才能被浏览器执行,它也区分了string和String这两个数据类型。通常来说,string表示原生类型,而String表示对象。 原生string JavaScript在ES6标准里支持6种原生类型(number),string是其中之一。 原生的string是不包含属性的值(即没有properties),包括字面上没有定义类型、字面上定义了string、字面上定义了String和一些从string函数调用返回的strings也都可以被归为原生类型: 以上三...
-
下一篇
开源软件出口管制合规政策与管控策略研究
目录 一、 概述 二、 各国针对开源软件的出口管制政策 (一)美国出口管制法规对于开源软件的管控 (二)中国、欧盟和日本出口管制法规对于开源软件的管控 三、 重要开源社区的出口管制策略 (一)国际开源组织项目出口管制政策 1. 开源基金会 2. 开源许可证 3. 代码托管平台 (二)社区声明对开源软件使用的影响 四、 开源软件出口管制合规管控实践 (一)开源软件应用管控 1.受EAR管辖开源软件识别 2.受EAR管辖开源软件备案 (二)开源对外贡献管控 1.自研代码贡献 2.开源项目建设 五、 开源软件管控改进措施和风险分析 (一)开源软件管控差距和改进 (二)开源软件潜在风险分析 六、参考文献 一:概述 开源软件是源代码可以任意获取的计算机软件。在获取开源软件源代码的基础上,任何人都能查看、修改和分发他们认为合适的代码。开源是信息社会的生产方式之一,特点是大维度的协作与网络服务化,全球范围的源代码开放,是开源软件的一个特征。我国的开源软件技术已经全面进军操作系统、云原生、人工智能、大数据等主要领域,例如在容器技术、微服务技术、DevOps技术上,均有中国的开源贡献。开源软件已在各行各...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Hadoop3单机部署,实现最简伪集群
- SpringBoot2全家桶,快速入门学习开发网站教程
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
- Dcoker安装(在线仓库),最新的服务器搭配容器使用
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- MySQL8.0.19开启GTID主从同步CentOS8
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- SpringBoot2整合Thymeleaf,官方推荐html解决方案