当前位置: 首页 > news >正文

LeetCode 1206. 实现跳表

不使用任何库函数,设计一个跳表。

跳表是在 O(log(n)) 时间内完成增加、删除、搜索操作的数据结构。跳表相比于树堆与红黑树,其功能与性能相当,并且跳表的代码长度相较下更短,其设计思想与链表相似。

例如,一个跳表包含 [30, 40, 50, 60, 70, 90],然后增加 80、45 到跳表中,以下图的方式操作:

 

跳表中有很多层,每一层是一个短的链表。在第一层的作用下,增加、删除和搜索操作的时间复杂度不超过 O(n)。跳表的每一个操作的平均时间复杂度是 O(log(n)),空间复杂度是 O(n)。

注意,跳表中可能存在多个相同的值,你的代码需要处理这种情况。

题目分析

跳表其实就是加了几层索引的链表,一共有 N 层,以 0 ~ N-1 层表示,设第 0 层是原链表,抽取其中部分元素,在第 1 层形成新的链表,上下层的相同元素之间连起来;再抽取第 1 层部分元素,构成第 2 层,以此类推。

为什么需要random函数呢?

需要某种手段来维护索引与原始链表大小之间的平衡,也就是说,如果链表中结点多了,索引结点就相应地增加一些,避免复杂度退化,以及查找、插入、删除操作性能下降。

如果你了解红黑树、AVL 树这样平衡二叉树,你就知道它们是通过左右旋的方式保持左右子树的大小平衡,而跳表是通过随机函数来维护前面提到的“平衡性”。

【查找】

跳表其实就是加了几层索引的链表,一共有 N 层,以 0 ~ N-1 层表示,设第 0 层是原链表,抽取其中部分元素,在第 1 层形成新的链表,上下层的相同元素之间连起来;再抽取第 1 层部分元素,构成第 2 层,以此类推。
具体选哪些元素呢?题目给的是coinFlip,也就是每次插入元素的时候,都进行一次 50% 概率的判断,如果 true 则向上层添加一个索引,如果 false 就不加了。

【添加和删除】

添加和删除的时候,存在数据重复的问题,经过我的测试,发现题目要求的是,将重复的数据当做不同的值来对待,即存了一个元素 9 ,再存一个 9 ,此时表里有俩 9 ,相互独立,删除一个 9 ,表里应该还剩余有一个9。

由于添加和删除元素都是从底层开始,而在查找的时候是从顶层开始查找的,因此可以使用一个数组记录每层的“跳跃节点”的位置,就不必反复的从顶层开始查找每层的位置了。

在添加的时候,首先是查找到添加位置,过程与查找类似,首先找到表中 >= 插入元素 的最小节点位置,然后插入节点, 50% 概率判别,如果需要添加索引,就添加索引,继续 50% 概率判别,直到结束。

在删除的时候,首先也是查找,找到表中所有 = 插入元素的节点位置(每层只找一个,找到直接跳层即可),然后挨个删除。

代码实现

class Skiplist {class SkipListNode {int val;int cnt;  // 当前val出现的次数SkipListNode[] levels;  // start from 0SkipListNode() {levels = new SkipListNode[MAX_LEVEL];}}private double p = 0.5;private int MAX_LEVEL = 16;private SkipListNode head;  // 头结点private int level;  // private Random random;public Skiplist() {// 保存此level有利于查询(以及其他操作)level = 0;  // 当前 skiplist的高度(所有数字 level数最大的)head = new SkipListNode();random = new Random();}// 返回target是否存在于跳表中public boolean search(int target) {SkipListNode curNode = head;for (int i = level-1; i >= 0; i--) {while (curNode.levels[i] != null && curNode.levels[i].val < target) {curNode = curNode.levels[i];}}curNode = curNode.levels[0];return (curNode != null && curNode.val == target);}// 插入一个元素到跳表。public void add(int num) {SkipListNode curNode = head;// 记录每层能访问的最右节点SkipListNode[] levelTails = new SkipListNode[MAX_LEVEL];for (int i = level-1; i >= 0; i--) {while (curNode.levels[i] != null && curNode.levels[i].val < num) {curNode = curNode.levels[i];}levelTails[i] = curNode;}curNode = curNode.levels[0];if (curNode != null && curNode.val == num) {// 已存在,cnt 加1curNode.cnt++;} else {// 插入int newLevel = randomLevel();if (newLevel > level) {for (int i = level; i < newLevel; i++) {levelTails[i] = head;}level = newLevel;}SkipListNode newNode = new SkipListNode();newNode.val = num;newNode.cnt = 1;for (int i = 0; i < level; i++) {newNode.levels[i] = levelTails[i].levels[i];levelTails[i].levels[i] = newNode;}}}private int randomLevel() {int level = 1;  // 注意思考此处为什么是 1 ?while (random.nextDouble() < p && level < MAX_LEVEL) {level++;}return level > MAX_LEVEL ? MAX_LEVEL : level;}// 在跳表中删除一个值,如果 num 不存在,直接返回false. 如果存在多个 num ,删除其中任意一个即可。public boolean erase(int num) {SkipListNode curNode = head;// 记录每层能访问的最右节点SkipListNode[] levelTails = new SkipListNode[MAX_LEVEL];for (int i = level-1; i >= 0; i--) {while (curNode.levels[i] != null && curNode.levels[i].val < num) {curNode = curNode.levels[i];}levelTails[i] = curNode;}curNode = curNode.levels[0];if (curNode != null && curNode.val == num) {if (curNode.cnt > 1) {curNode.cnt--;return true;}// 存在,删除for (int i = 0; i < level; i++) {if (levelTails[i].levels[i] != curNode) {break;}levelTails[i].levels[i] = curNode.levels[i];}while (level > 0 && head.levels[level-1] == null) {level--;}return true;} return false;}
}

代码实现二

class Skiplist {int MAX_LEVEL = 16;int curLevel;Node head;public Skiplist() {curLevel = 1;head = new Node(-1);head.next_point = new Node[MAX_LEVEL];}public boolean search(int target) {Node temp = head;//从最顶层索引找for (int i = MAX_LEVEL - 1; i >=0; i--) {while (temp.next_point[i] != null && temp.next_point[i].val <= target) {if (temp.next_point[i].val == target)return true;elsetemp = temp.next_point[i];}}// 判断temp的下个节点是否满足条件if (temp.next_point[0] != null && temp.next_point[0].val == target)return true;return false;}public void add(int num) {int level = randomLevel(0.5);if (level > curLevel)curLevel = level;Node node = new Node(num);node.next_point = new Node[level];Node[] forward = new Node[level];Arrays.fill(forward, head);Node temp = head;// 找到前驱节点for (int i = level - 1; i >= 0; i--) {while (temp.next_point[i] != null && temp.next_point[i].val < num)temp = temp.next_point[i];forward[i] = temp;}// 更新连接for (int i = 0; i < level; i++) {node.next_point[i] = forward[i].next_point[i];forward[i].next_point[i] = node;}}public boolean erase(int num) {Node[] forward = new Node[curLevel];Node temp = head;for (int i = curLevel - 1; i >= 0; i--) {while (temp.next_point[i] != null && temp.next_point[i].val < num)temp = temp.next_point[i];forward[i] = temp;}boolean res = false;if (temp.next_point[0] != null && temp.next_point[0].val == num) {res = true;// 更新连接for (int i = curLevel - 1; i >= 0; i--)if (forward[i].next_point[i] != null && forward[i].next_point[i].val == num)forward[i].next_point[i] = forward[i].next_point[i].next_point[i];}// 删除孤立节点层while (curLevel > 1 && head.next_point[curLevel - 1] == null)curLevel -= 1;return res;}public int randomLevel(double p) {int level = 1;while (Math.random() < p && level < MAX_LEVEL)level++;return level;}
}class Node {int val;Node[] next_point;public Node(int val) {this.val = val;}
}

相关文章:

LeetCode 1206. 实现跳表

不使用任何库函数&#xff0c;设计一个跳表。 跳表是在 O(log(n)) 时间内完成增加、删除、搜索操作的数据结构。跳表相比于树堆与红黑树&#xff0c;其功能与性能相当&#xff0c;并且跳表的代码长度相较下更短&#xff0c;其设计思想与链表相似。 例如&#xff0c;一个跳表包…...

离散数学_九章:关系(2)

9.2 n元关系及其应用 1、n元关系&#xff0c;关系的域&#xff0c;关系的阶2、数据库和关系 1. 数据库 2. 主键 3. 复合主键 3、n元关系的运算 1. 选择运算 (Select) 2. 投影运算 (Project) 3. 连接运算 (Join) n元关系&#xff1a;两个以上集合的元素间的关系 1、n元关系…...

[ubuntu][原创]通过apt方式去安装libnccl库

ubuntu18.04版本安装流程&#xff1a; wget https://developer.download.nvidia.com/compute/cuda/repos/ubuntu1804/x86_64/cuda-ubuntu1804.pin sudo mv cuda-ubuntu1804.pin /etc/apt/preferences.d/cuda-repository-pin-600 sudo apt-key adv --fetch-keys https://develo…...

YonLinker连接集成平台构建新一代产业互联根基

近日&#xff0c;由用友公司主办的“2023用友BIP技术大会“在用友产业园&#xff08;北京&#xff09;盛大召开&#xff0c;用友介绍了更懂企业业务的用友BIP-iuap平台&#xff0c;并发布了全面数智化能力体系&#xff0c;助力企业升级数智化底座&#xff0c;加强加速数智化推进…...

泛型的详解

泛型的理解和好处 首先我们先来看看泛型的好处 1)编译时&#xff0c;检查添加元素的类型&#xff0c;提高了安全性 2)减少了类型转换的次数&#xff0c;提高效率[说明] 不使用泛型 Dog -> Object -> Dog//放入到ArrayList 会先转成Object&#xff0c;在取出时&#x…...

用科技创造未来!流辰信息技术助您实现高效办公

随着社会的迅猛发展&#xff0c;科技的力量无处不见。它正在悄悄地改变整个社会&#xff0c;让人类变得进步和文明&#xff0c;让生活变得便捷和高效。在办公自动化强劲发展的今天&#xff0c;流辰信息技术让通信业、电网、汽车、物流等领域的企业实现了高效办公&#xff0c;数…...

基于R语言APSIM模型

随着数字农业和智慧农业的发展&#xff0c;基于过程的农业生产系统模型在模拟作物对气候变化的响应与适应、农田管理优化、作物品种和株型筛选、农田固碳和温室气体排放等领域扮演着越来越重要的作用。 APSIM (Agricultural Production Systems sIMulator)模型是世界知名的作物…...

块状链表实现BigString大字符串操作(golang)

前言 块状链表是介于链表和数组之间的数据结构&#xff0c;能够在 O ( n ) O(\sqrt{n}) O(n ​)时间内完成插入、删除、访问操作。 数据结构如图所示。假设最大容量为 n n n, 则它有一个长度为 s n s\sqrt{n} sn ​的链表。链表中每个结点是一个长度为 2 n 2 \times \sqrt{…...

项目问题记录(持续更新)

1.在 yarn install的时候报 error achrinza/node-ipc9.2.2: The engine "node" is incompatible with this module. Expected version "8 || 10 || 12 || 14 || 16 || 17". Got "20.1.0" error Found incompatible module.需要执行 yarn config…...

Linux的进程

目录 一、进程占用的内存资源 二、进程的系统环境 三、进程一直在切换 四、父进程和子进程 五、进程状态 六、查看进程 1.ps -ef 列出所有进程 2.ps -lax 列出所有进程 3.ps aux列出所有进程 4.树形列出所有进程 七、作业&#xff08;用来查看管理进程&#xff09; …...

与其焦虑被 AI 取代或猜测前端是否已死, 不如看看 vertical-align 扎实你的基础!!!

与其焦虑被 AI 取代或猜测前端是否已死, 不如看看 vertical-align 扎实你的基础!!! vertical-align 设置 display 值为 inline, inline-block 和 table-cell 的元素竖直对齐方式. 从 line-height: normal 究竟是多高说起 我们先来看一段代码, 分析一下为什么第二行的行高, 也就…...

路由、交换机、集线器、DNS服务器、广域网/局域网、端口、MTU

前言&#xff1a;网络名词术语解析(自行阅读扫盲)&#xff0c;推荐大家去读户根勤的《网络是怎样连接的》 路由(route)&#xff1a; 数据包从源地址到目的地址所经过的路径&#xff0c;由一系列路由节点组成。某个路由节点为数据包选择投递方向的选路过程。 路由器工作原理 路…...

在全志V851S开发板上进行屏幕触摸适配

1.修改屏幕驱动 从ft6236 &#xff08;删掉&#xff0c;不要保留&#xff09;&#xff0c;改为下面的 路径&#xff1a;/home/wells/tina-v853-open/tina-v853-open/device/config/chips/v851s/configs/lizard/board.dts&#xff08;注意路径&#xff0c;要设置为自己的实际路…...

字符串拷贝时的内存重叠问题

字符串拷贝时的内存重叠问题 1.什么是内存重叠 拷贝的目的地址在源地址的范围内&#xff0c;有重叠。 如在写程序的过程中&#xff0c;我们用到的strcpy这个拷贝函数&#xff0c;在这个函数中我们定义一个目的地址&#xff0c;一个源地址&#xff0c;在拷贝的过程中如果内存重…...

告别PPT手残党!这6款AI神器,让你秒变PPT王者!

如果你是一个PPT手残党&#xff0c;每每制作PPT总是让你焦头烂额&#xff0c;那么你一定需要这篇幽默拉风的推广文案&#xff01; 我向你保证&#xff0c;这篇文案将帮助你发现6款AI自动生成PPT的神器&#xff0c;让你告别PPT手残党的身份&#xff0c;成为一名PPT王者。 无论…...

JVM配置与优化

参考&#xff1a; JVM内存分区及作用&#xff08;JDK8&#xff09; https://blog.csdn.net/BigBug_500/article/details/104734957 java 进程占用系统内存过高分析 https://blog.csdn.net/fxh13579/article/details/104754340 Java之jvm和线程的内存 https://blog.csdn.ne…...

电力系统储能调峰、调频模型研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…...

C++基础之类、对象一(类的定义,作用域、this指针)

目录 面向对象的编程 类的引入 简介 类的定义 简介 访问限定符 命名规则 封装 简介 类的作用域 类的大小及存储模型 this指针 简介 面向对象的编程 C与C语言不同&#xff0c;C是面向对象的编程&#xff0c;那么什么是面向对象的编程呢&#xff1f; C语言编程&#xff0c;规定…...

javaScript---设计模式-封装与对象

目录 1、封装对象时的设计模式 2、基本结构与应用示例 2.1 工厂模式 2.2 建造者模式 2.3 单例模式 封装的目的&#xff1a;①定义变量不会污染外部&#xff1b;②能作为一个模块调用&#xff1b;③遵循开闭原则。 好的封装&#xff08;不可见、留接口&#xff09;&#xff1a;①…...

【消息中间件】kafka高性能设计之内存池

文章目录 前言实现创建内存池分配内存释放内存 总结 前言 Kafka的内存池是一个用于管理内存分配的缓存区域。它通过在内存上保留一块固定大小的内存池&#xff0c;用于分配消息缓存、批处理缓存等对象&#xff0c;以减少频繁调用内存分配函数的开销。 Kafka内存池的实现利用了…...

创建型模式——单例(singleton)

1. 模式说明 单例模式保证类只有一个实例&#xff1b;创建一个对象&#xff0c;当你创建第二个对象的时候&#xff0c;此时你获取到的是已经创建过的对象&#xff0c;而不是一个新的对象&#xff1b; 1.1 使用场景 共享资源的访问权限&#xff1b;任务的管理类&#xff1b;数…...

算法:迷宫问题

描述 定义一个二维数组 N*M &#xff0c;如 5 5 数组下所示&#xff1a; int maze[5][5] { 0, 1, 0, 0, 0, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 0, 0, 0, 0, 1, 0, }; 它表示一个迷宫&#xff0c;其中的1表示墙壁&#xff0c;0表示可以走的路&#xff0c;只能横着走或…...

聊聊并发编程的12种业务场景

前言 并发编程是一项非常重要的技术&#xff0c;无论在面试&#xff0c;还是工作中出现的频率非常高。 并发编程说白了就是多线程编程&#xff0c;但多线程一定比单线程效率更高&#xff1f; 答&#xff1a;不一定&#xff0c;要看具体业务场景。 毕竟如果使用了多线程&…...

MySQL执行顺序

MySQL执行顺序 MySQL语句的执行顺序也是在面试过程中经常问到的问题&#xff0c;并且熟悉执行顺序也有助于SQL语句的编写。 SELECT FROM JOIN ON WHERE GROUP BY HAVING ORDER BY LIMIT执行顺序如下&#xff1a; FROM ON JOIN WHERE GROUP BY # (开始使用别名) SUM # SUM等…...

引领真无线耳机未来趋势,NANK南卡OE骨传导真无线耳机惊艳亮相

传统的蓝牙耳机存在很多问题&#xff0c;例如续航时间短、长期佩戴耳朵会不舒服&#xff0c;甚至影响听力等等。为了解决这些问题&#xff0c;在骨传导领域深耕十多年的南卡品牌推出了这款真无线骨传导耳机——NANK南卡 OE。 NANK南卡OE即将正式上线&#xff0c;这一消息一经宣…...

5款写作神器,帮助你写出5w+爆款文案,好用到哭

我不允许还有文案小白、新手博主不知道这5款写作利器&#xff01; 每次一写文案就头秃的新媒体工作者&#xff0c;赶紧看过来吧&#xff01;这5款好用到爆的写作神器&#xff0c;喝一杯咖啡的时间就能完成写作。 我和同事都是用它们&#xff0c;出了很多的爆款&#xff0c;现…...

相交链表问题

给你两个单链表的头节点 headA 和 headB &#xff0c;请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点&#xff0c;返回 null 。 图示两个链表在节点 c1 开始相交&#xff1a; 题目数据 保证 整个链式结构中不存在环。 注意&#xff0c;函数返回结果后&…...

[ubuntu] ax200网卡虚接,导致系统根目录占满而无法进入系统的奇葩问题

20230508&#xff0c;我像往常一样,打开电脑发现根目录满了&#xff0c;报警了&#xff0c;所以按照网上的教程&#xff0c;清理了一下根目录的文件&#xff0c;没想到背后是网卡问题… 文章目录 1.进入终端模式2.查看占用情况3.清理系统log文件3.1 清理/var/log/syslog3.2 清…...

本地字体库的引入方法

本地字体库是指在计算机系统中存储的一组字体文件&#xff0c;通常包含多种字体格式&#xff0c;如TTF、OTF、WOFF等。引入本地字体库可以让用户在使用计算机时可以选择不同的字体&#xff0c;从而提高用户的使用体验。 本地字体库的引入方式有多种&#xff0c;其中比较常用的是…...

7种优秀的导航菜单设计总结

导航是应用程序界面中最常见的模块之一&#xff0c;在链接应用程序中起着每个页面的作用。 不同的设计需求和业务目标决定了导航的设计因品而异&#xff0c;移动设备的尺寸远小于计算机。因此&#xff0c;在设计移动终端导航时&#xff0c;应考虑更全面&#xff0c;以确保简单…...

星月教你做网站回顾文档/百度推广售后客服电话

***服务大家都知道&#xff0c;虚拟专用网络&#xff0c;今天我们就来说说怎么用linux搭建pptp *** 说实话&#xff0c;我这也算是第一次照着命令敲然后成功了的&#xff0c;最后才搞明白怎么回事&#xff0c;特贡献出来给大家&#xff0c;希望大家共同进步 系统环境: 2.6.18-9…...

网站备案核实/seo日常工作内容

更新范围 文档页眉页脚设置文档标题与文件名称保持一致。段落格式文档字体统一。(通常是宋体)文档字体: 大小要求,行距要求表格,自适应纸张大小。(待完善)文档目录更新。删除文档作者信息。设置文档仅查看。检查清单2方案3其他问题文档页眉页脚设置 页眉:不同的章节显…...

p2p网站制作郑州/百度推广没有一点效果

我在绘制简单的折线图时遇到了问题&#xff0c;在y轴上有数值&#xff0c;其中轴上的日期是&#xff1a;我的python代码如下&#xff1a;i 0 #iterator for weather dataxAxis []yAxis []for trainingDate in observations[:,0]: #getting values in x and y axis for the d…...

招工做哪个网站/百度大数据预测平台

深入C系列&#xff1a; 1、《C STL中文版》 2、《More Effective C&#xff08;中文版&#xff09;》 3、《深度探索C对象模型》 4、《泛型编程与STL》 5、《Effective STL》 6、《C Primer中文版》 7、《C程序设计原理与实践》 8、《C编程思想》 9、《C编程规范&…...

品牌网站设计案例/怎样把自己的产品放到网上销售

扩展就是向一个已有的类、结构体或枚举类型添加新功能&#xff08;functionality&#xff09;。这包括在没有权限获取原始源代码的情况下扩展类型的能力&#xff08;即逆向建模&#xff09;。扩展和 Objective-C 中的分类&#xff08;categories&#xff09;类似。&#xff08;…...

上海cms模板建站/销售策略和营销策略

所谓隔代通信就是A 与C的通信 A -> B -> C代码实例 A.vue <template><div id"app"><!-- 此处监听了事件&#xff0c;可以在C组件中直接触发 --><b-childnameToB"nameToB" nameToC"nameToC" buttonClick"butto…...