数据结构-跳表:自己理解的通俗总结 + 使用python实现

前提:有序数据 + 基于链表
操作:有序数据的查询 + 插入 + 删除
时间复杂度:O(logn)
思想:将有序数据进行分层,最底层是存储所有的数据,之后每一层逐层减少数据,并且需要注意 每一层的数据是基于其前一层的数据随机而来。(这里有一个点需要注意:这种随机可能会造成上层的数据失控,因此是实现时一般会做一定的 数据数量限制的伪随机 / 动态的变化概率 防止出现数据失控,造成数据太少或太多)

举例
现有数据(设定总共 3 层):1 -> 2 -> 5 -> 7 ->14 -> 23 -> 47
则分层完成后结果如下(每一层的每一个值是通过设定的概率筛选出来的):
第 0 层:1 -> 2 -> 5 -> 7 ->14 -> 23 -> 47
第 1 层:1 -> 5 -> 7 -> 23
第 2 层:1 -> 23

  • 查询逻辑:每次查询都从顶层开始查询,并且降层时是从本层现校验的元素开始筛选而不是从头开始。
    现在查询 14:
    -1-:首先查询第 2 层,发现比 1 大比 23 小,开始查询第 1 层
    -2-:从 1 开始查询第 1 层,发现比最后一个值 23 小,开始查询第 0 层
    -3-:从 7 开始查询第 0 层,发现 7 的下一个值就是 14,查询完成
  • 插入数据逻辑:每次从最底层开始插入
    首先插入第 0 层对应的位置,然后根据设定的概率判断是否要升层,如果升层则每一层都做同样的判断直到到达最顶层或者不需要升层为止。
  • 删除数据逻辑:从顶层开始逐层查询元素删除
    首先从第 2 层开始查询,如果查询到了则删除,然后依次降层查询删除。

实现
接下来用python3来实现一下跳表(该实现没做伪随机,有一定概率出现上层数据失控):

import randomclass SkipListNode:def __init__(self, key, level):self.key = keyself.forward = [None] * (level + 1)class SkipList:def __init__(self, max_level, p):self.max_level = max_level  # 最大层数self.p = p  # 向上提升一个级别的概率self.header = SkipListNode(None, max_level)  # 创建头节点self.level = 0  # 当前最高层数def random_level(self):level = 0while random.random() < self.p and level < self.max_level:level += 1return leveldef insert(self, key):update = [None] * (self.max_level + 1)current = self.header# 从最高层开始,找到每层中小于插入键的最大键for i in range(self.level, -1, -1):while current.forward[i] and current.forward[i].key < key:current = current.forward[i]update[i] = current# 移动到下一层current = current.forward[0]# 如果当前层没有该键,则插入if current is None or current.key != key:# 随机确定节点高度rlevel = self.random_level()# 如果新节点的层数更高if rlevel > self.level:for i in range(self.level + 1, rlevel + 1):update[i] = self.headerself.level = rlevel# 创建新节点n = SkipListNode(key, rlevel)for i in range(rlevel + 1):n.forward[i] = update[i].forward[i]update[i].forward[i] = ndef search(self, key):current = self.headerfor i in range(self.level, -1, -1):while current.forward[i] and current.forward[i].key < key:current = current.forward[i]current = current.forward[0]if current and current.key == key:return Truereturn Falsedef display(self):print("Skip List")for i in range(self.level + 1):print("Level {}: ".format(i), end="")node = self.header.forward[i]while node:print(node.key, end=" ")node = node.forward[i]print("")# 创建跳表实例,最大层设为3,提升概率为0.5(这个概率随便设置)
skip_list = SkipList(3, 0.5)
skip_list.insert(3)
skip_list.insert(6)
skip_list.insert(7)
skip_list.insert(9)
skip_list.insert(12)
skip_list.insert(19)
skip_list.insert(17)# 显示跳表
skip_list.display()# 搜索元素
print("Search for 6:", skip_list.search(6))
print("Search for 15:", skip_list.search(15))

本文完~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/11469.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

软件文档-总体测试计划书(Word原件2024)

软件资料清单列表部分文档&#xff1a; 工作安排任务书&#xff0c;可行性分析报告&#xff0c;立项申请审批表&#xff0c;产品需求规格说明书&#xff0c;需求调研计划&#xff0c;用户需求调查单&#xff0c;用户需求说明书&#xff0c;概要设计说明书&#xff0c;技术解决…

Spring事件分析以及多种使用方式实践 使用场景 附可执行demo

我们先说说它的特点&#xff0c;优缺点&#xff0c;以及使用场景&#xff0c;然后再说具体是怎么做的 Spring事件驱动的优点 松耦合 事件驱动模型通过发布-订阅机制促进组件间的解耦&#xff0c;发送者和接收者不需要直接知道对方的存在&#xff0c;只需关注事件本身&#xff…

【机器学习-06】Scikit-Learn机器学习工具包进阶指南:机器学习分类模型实战与数据可视化分析

&#x1f3a9; 欢迎来到技术探索的奇幻世界&#x1f468;‍&#x1f4bb; &#x1f4dc; 个人主页&#xff1a;一伦明悦-CSDN博客 ✍&#x1f3fb; 作者简介&#xff1a; C软件开发、Python机器学习爱好者 &#x1f5e3;️ 互动与支持&#xff1a;&#x1f4ac;评论 &…

[猫头虎分享21天微信小程序基础入门教程]第7天:小程序的权限与API使用

第7天&#xff1a;小程序的权限与API使用 &#x1f510; 自我介绍 大家好&#xff0c;我是猫头虎&#xff0c;一名全栈软件工程师。今天我们将继续微信小程序的学习&#xff0c;重点了解如何使用微信小程序的API&#xff0c;以及如何管理和请求小程序的权限。通过这些知识&am…

可视化:智慧能源解决方案,降本增效,运筹帷幄。

智慧能源可视化解决方案是一种利用先进的技术和工具&#xff0c;将能源数据以直观、可视的方式呈现出来&#xff0c;帮助企业更好地管理能源使用&#xff0c;降低成本&#xff0c;提高效率的解决方案。 以下是一些智慧能源可视化解决方案可以帮助企业降本增效、智连未来的方式&…

js前端获取农历日期

对于公历来说&#xff0c;直接 new 一个 Date 就能获取到&#xff0c;而对于农历来讲可就很难了&#xff0c;因为农历需要有许多复杂计算&#xff0c;虽然一般用的甚少&#xff0c;但对于某些场景来说还是会需要的&#xff0c;那么怎样获取农历日期呢&#xff1f; 这里推荐一个…

Linux:进程信号

生活角度的信号 a.信号在生活中&#xff0c;随时可以产生(信号的产生和我是异步的) b.你能认识这个信号 c.我们知道信号产生了&#xff0c;我能识别这个信号&#xff0c;信号该怎么处理 d.我们可能正在做着更重要的事情&#xff0c;把到来的信号暂不处理(1.我记得这个事 2.…

这是摆脱困境的最好方法

20多年前&#xff0c;我开始涉足创业&#xff0c;经历过的那种停滞感我都记不清了。这是这条职业道路上最常见的挣扎之一&#xff0c;而且很难摆脱。 卡住的城市是一个地方&#xff0c;任何有创造力的&#xff0c;自由职业者和好奇的人经常去。这是一个很难逃离的地方。 被困…

要不还是别搞Google Play了

​好消息&#xff1a;误导性条款合规了 坏消息: 应用被暂停了&#xff0c;太难玩了 5.13日收到邮件&#xff0c;被告知应用因为应用内体验问题被暂停上架了 邮件大意是由于违反执行流程政策(Violation of Enforcement Process policy)。 这个政策主要是讲Google Play会对账户/…

求四个整数中的最大值(函数)(C语言)

一、N-S流程图&#xff1b; 二、运行结果&#xff1b; 三、源代码&#xff1b; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h>int main() {//初始化变量值&#xff1b;int a, b, c, d, max;//获取用户输入的数据&#xff1b;printf("请输入4个整数&#x…

你知道在 TS 中判断两个类型相等有多难吗?

公众号&#xff1a;程序员白特&#xff0c;欢迎一起交流学习~ TypeScript 中的类型相等 如果我们想判断两个变量是否相等&#xff0c;可以简单的通过 或 来进行比较&#xff0c;但是对比两个类型则不行。 在 TypeScript 中&#xff0c;类型是静态的&#xff0c;只会在编译时…

【Esp32S3 | Arduino】在Arduino中使用C++的高级特性

文章目录 前言一、Arduino中的Vector示例代码二、Arduino中的Map示例代码前言 最近在玩Arduino,自上次发现Arduino可以用Template,能使用高级宏后,这次发现Arduino竟可以使用C++中的一些STL容器,这属实令人震惊。起因是我打算做一个动态的数组,但是手动实现一些操作属实麻烦…

polars学习-03 数据类型转换

背景 polars学习系列文章&#xff0c;第3篇 数据类型转换。 该系列文章会分享到github&#xff0c;大家可以去下载jupyter文件 仓库地址&#xff1a;https://github.com/DataShare-duo/polars_learn 小编运行环境 import sysprint(python 版本&#xff1a;,sys.version.spli…

Hack The Box-SolarLab

总体思路 SMB获取敏感信息->CVE-2023-33733漏洞注入->CVE-2023-32315->敏感信息泄露 信息收集&端口利用 nmap -sSVC -p1-10000 10.10.11.16发现目标开放了80、135、139、445和6791端口&#xff0c;并且对应的端口也给出了重定向的标志&#xff0c;将域名加入到…

实验过程演示【计算机网络实验】

前言 这是陈旧已久的草稿2023-05-20 11:23:54 这个是计算机网络的一个实验&#xff0c;现在也不知道这个是啥来着。 现在2024-5-12 22:33:17&#xff0c;发布到[计算机网络实验]专栏中。 实验过程演示 2023-5-18 20:17:45 1&#xff0e;搭建一个多跳网络拓扑&#xff0c;…

算法题解记录25+++验证二叉搜索树(百日筑基)

题目描述&#xff1a; 难度&#xff1a;中等 给你一个二叉树的根节点 root &#xff0c;判断其是否是一个有效的二叉搜索树。 有效 二叉搜索树定义如下&#xff1a; 节点的左 子树 只包含 小于 当前节点的数。节点的右子树只包含 大于 当前节点的数。所有左子树和右子树自身必…

软件开发模型介绍

软件开发模型&#xff08;Software Development Model&#xff09;是指软件开发全部过程、活动和任务的结构框架。它清晰、直观地表达软件开发全过程&#xff0c;明确规定了要完成的主要活动和任务&#xff0c;用来作为软件项目工作的基础。 一、常见的软件开发模型&#xff1…

【MISRA-C-2012】:标准的理解与学习

标准的理解与学习 引用二、Misra-C 规则Misra-C全解读 - Rule 1 标准的C语言环境&#xff08;待更新&#xff09;Misra-C全解读 - Rule 2 未使用的代码&#xff08;待更新&#xff09;Misra-C全解读 - Rule 3 注释&#xff08;待更新&#xff09;Misra-C全解读 - Rule 4 字符与…

ThinkPHP+MySQL查询数据的时候计算两个经纬度之间的距离并根据距离进行筛选

原需求实现说明 新增了一个按距离进行筛选的需求。需要把查询代码做如下修改 /*** 求职意向* return void* throws \think\exception\DbException*/public function get_lists(){$request $this->request->get();if(empty($request[lng]) || empty($request[lat])){$th…

如何抠图?6个简单方便的抠图软件教你自己快速抠图

如何抠图&#xff1f;6个简单方便的抠图软件教你自己快速抠图 抠图是图像处理中常见的操作之一&#xff0c;它可以帮助我们从一幅图像中抠出特定的部分&#xff0c;通常用于制作合成图、更换背景或修改图像内容。下面介绍的6款简单方便的抠图软件可以帮助您快速进行抠图操作&a…