高效的跳表

2024-02-02 12:44
文章标签 高效 跳表

本文主要是介绍高效的跳表,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

高效的跳表

  • 一、 概念
  • 二、 实现原理
  • 三、存在的问题
  • 四、解决方法
  • 五、如何保证效率
  • 六、代码实现
  • 七、总结
    • 对比平衡搜索树
    • 对比哈希表

一、 概念

跳表,是一种用来查询数据的数据结构,它是由William Pugh发明的,借助有序链表,来实现高效地查询

二、 实现原理

William Pugh的优化思路是,每相邻两个节点升高一层,增加一个指针,让指针指向下下个节点,依次类推,形成下图的c,每一层都能排除一半,类似于二分查找,时间复杂度为O(log n)
在这里插入图片描述

三、存在的问题

插入删除数据时,就会打乱上下相邻两层链表上节点个数严格的2:1的对应关系,而要维持这种对应关系,就必须把新插入的节点及其后面的所有节点重新进行调整,但这会让时间复制度退化为O(n)

四、解决方法

William Pugh为了解决这个问题,做了一个大胆的处理,不再严格要求这种对应比例关系,而是插入一个节点的时候,随机出一个层数,这样每次插入和删除都不再需要考虑其它节点的层数,实现起来也就简单多了
在这里插入图片描述

五、如何保证效率

一般跳表会设计一个最大层数maxLevel的限制,其次,会设置一个多增加一层的概率p,伪代码如下图
在这里插入图片描述
在Redis的skiplist实现中,p取1/4,maxLevel取32,根据前面的伪代码,定量分析如下图
在这里插入图片描述
一个节点的平均层数,计算如下图
在这里插入图片描述

六、代码实现

首先,得定义一个节点结构体,一个节点由值和它的层数(下一个结点指针的集合)组成
在这里插入图片描述
跳表,则有三个成员,_head是一个哨兵节点,概率p和最大层数maxLevel给缺省值即可,同时,构造函数内,还需给一个随机数种子,后面用来获取节点层数
在这里插入图片描述
在这里插入图片描述
查找某个值,从上往下开始找,让当前节点指针指向哨兵节点,判断它的下一个节点值和目标值的大小关系,下一个节点值为空或比目标值大,就往下走,即层数下标-1;如果比目标值小,就往右走;否则,则表明找到了,直接返回true即可,如果层数下标都小于0了,就返回false,表明没有找到
在这里插入图片描述
类似于单链表,跳表要插入一个节点或删除一个节点,就必须找到目标节点的前一个节点的集合,这样才能插入或者删除节点,例如,插入20,它的前一个节点集合,自上而下为_head,17,19,如下图
在这里插入图片描述
代码类似于查找目标值,不再赘述
在这里插入图片描述
获取节点层数
在这里插入图片描述

插入节点,首先,获取它的前一个节点的集合,其次,获取节点的层数,同时,需要保证_head节点的层数始终是所有节点中最多的,然后就是链表的插入操作了
在这里插入图片描述
删除节点,首先,得判断节点在不在,不在,就不需要删除了,直接返回即可,删除操作也就是链表的删除操作,最后,做一个层数更新,因为删除节点后,头节点的层数可能会有很多是不必要的,影响查找效率,所以做了一个小的优化
在这里插入图片描述
下面的一个leetcode题目,可以用来验证写的对不对
跳表

七、总结

对比平衡搜索树

对比AVL树和红黑树,都可以做到遍历数据有序,时间复杂度也差不多,而跳表的优势在于:

实现简单,容易控制,而平衡树增删改查都更复杂

跳表的额外空间消耗更低,平衡树节点需要存储三个节点指针,父亲和左右孩子,还有平衡因子或颜色等消耗

对比哈希表

相对于哈希表,优势小了很多,首先,哈希表查找数据的时间复杂度为O(1),比跳表快,跳表的优势在于:

遍历数据有序

跳表的空间消耗略小一些,哈希表存在链接指针和表空间消耗

哈希表扩容有性能损耗

哈希表在极端场景下,哈希冲突高,效率下降厉害,需要红黑树来弥补

这篇关于高效的跳表的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java高效实现PowerPoint转PDF的示例详解

《Java高效实现PowerPoint转PDF的示例详解》在日常开发或办公场景中,经常需要将PowerPoint演示文稿(PPT/PPTX)转换为PDF,本文将介绍从基础转换到高级设置的多种用法,大家... 目录为什么要将 PowerPoint 转换为 PDF安装 Spire.Presentation fo

C#使用Spire.Doc for .NET实现HTML转Word的高效方案

《C#使用Spire.Docfor.NET实现HTML转Word的高效方案》在Web开发中,HTML内容的生成与处理是高频需求,然而,当用户需要将HTML页面或动态生成的HTML字符串转换为Wor... 目录引言一、html转Word的典型场景与挑战二、用 Spire.Doc 实现 HTML 转 Word1

Redis实现高效内存管理的示例代码

《Redis实现高效内存管理的示例代码》Redis内存管理是其核心功能之一,为了高效地利用内存,Redis采用了多种技术和策略,如优化的数据结构、内存分配策略、内存回收、数据压缩等,下面就来详细的介绍... 目录1. 内存分配策略jemalloc 的使用2. 数据压缩和编码ziplist示例代码3. 优化的

使用SpringBoot+InfluxDB实现高效数据存储与查询

《使用SpringBoot+InfluxDB实现高效数据存储与查询》InfluxDB是一个开源的时间序列数据库,特别适合处理带有时间戳的监控数据、指标数据等,下面详细介绍如何在SpringBoot项目... 目录1、项目介绍2、 InfluxDB 介绍3、Spring Boot 配置 InfluxDB4、I

C#高效实现Word文档内容查找与替换的6种方法

《C#高效实现Word文档内容查找与替换的6种方法》在日常文档处理工作中,尤其是面对大型Word文档时,手动查找、替换文本往往既耗时又容易出错,本文整理了C#查找与替换Word内容的6种方法,大家可以... 目录环境准备方法一:查找文本并替换为新文本方法二:使用正则表达式查找并替换文本方法三:将文本替换为图

Python如何实现高效的文件/目录比较

《Python如何实现高效的文件/目录比较》在系统维护、数据同步或版本控制场景中,我们经常需要比较两个目录的差异,本文将分享一下如何用Python实现高效的文件/目录比较,并灵活处理排除规则,希望对大... 目录案例一:基础目录比较与排除实现案例二:高性能大文件比较案例三:跨平台路径处理案例四:可视化差异报

Java整合Protocol Buffers实现高效数据序列化实践

《Java整合ProtocolBuffers实现高效数据序列化实践》ProtocolBuffers是Google开发的一种语言中立、平台中立、可扩展的结构化数据序列化机制,类似于XML但更小、更快... 目录一、Protocol Buffers简介1.1 什么是Protocol Buffers1.2 Pro

Java高效实现Word转PDF的完整指南

《Java高效实现Word转PDF的完整指南》这篇文章主要为大家详细介绍了如何用Spire.DocforJava库实现Word到PDF文档的快速转换,并解析其转换选项的灵活配置技巧,希望对大家有所帮助... 目录方法一:三步实现核心功能方法二:高级选项配置性能优化建议方法补充ASPose 实现方案Libre

在IntelliJ IDEA中高效运行与调试Spring Boot项目的实战步骤

《在IntelliJIDEA中高效运行与调试SpringBoot项目的实战步骤》本章详解SpringBoot项目导入IntelliJIDEA的流程,教授运行与调试技巧,包括断点设置与变量查看,奠定... 目录引言:为良驹配上好鞍一、为何选择IntelliJ IDEA?二、实战:导入并运行你的第一个项目步骤1

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处