[C++][数据结构][跳表]详细讲解

目录

  • 0.什么是跳表?
  • 1.SkipList的优化思路
  • 2.SkipList的效率如何保证?
  • 3.SkipList实现
  • 4.SkipList VS 平衡搜索树 && Hash


0.什么是跳表?

  • SkipList本质上也是一种查找结构,用于解决算法中的查找问题,跟平衡搜索树和哈希表的价值是一样的,可以作为key或者key/value的查找模型
  • SkipList,是在有序链表的基础上发展起来的
    • 如果是一个有序的链表,查找数据的时间复杂度是 O ( N ) O(N) O(N)

1.SkipList的优化思路

  • 假如每相邻两个节点升高一层,增加一个指针,让指针指向下下个节点,如下图b所示

    • 这样所有新增加的指针连成了一个新的链表,但它包含的节点个数只有原来的一半
    • 由于新增加的指针,我们不再需要与链表中每个节点逐个进行比较了,需要比较的节点数大概只有原来的一半
  • 以此类推,可以在第二层新产生的链表上,继续为每相邻的两个节点升高一层,增加一个指针,从而产生第三层链表,这样搜索效率就进一步提高了,如下图c

  • SkipList正是受这种多层链表的想法的启发而设计出来的

  • 实际上,按照上面生成链表的方式,上面每一层链表的节点个数,是下面一层的节点个数的一半,这样查找过程就非常类似二分查找,使得查找的时间复杂度可以降低到 O ( l o g 2 N ) O(log_2N) O(log2N)

  • 但是这个结构在插入删除数据的时候有很大的问题

    • 插入或者删除一个节点之后,就会打乱上下相邻两层链表上节点个数严格的2:1的对应关系

    • 如果要维持这种对应关系,就必须把新插入的节点后面的所有节点(也包括新插入的节点)重新进行调整,这会让时间复杂度重新蜕化成O(n)

      请添加图片描述

  • SkipList的设计为了避免这种问题,做了一个大胆的处理

    • 不再严格要求对应比例关系,而是插入一个节点的时候随机出一个层数

    • 这样每次插入和删除都不需要考虑其他节点的层数, 这样就好处理多了

      请添加图片描述


2.SkipList的效率如何保证?

  • SkipList插入一个节点时随机出一个层数,听起来这么随意,如何保证搜索时的效率呢?

  • 首先要细节分析的是这个随机层数是怎么来的

    • 一般跳表会设计一个最大层数maxLevel的限制
    • 其次会设置一个多增加一层的概率p
    • 计算这个随机层数的伪代码如下图:
      请添加图片描述
  • **参考:**在Redis的SkipList实现中,这两个参数的取值为:

    p = 1/4;
    maxLevel = 32;
    
  • 根据前面randomLevel()的伪代码,产生越高的节点层数,概率越低

    • 节点层数至少为1,而大于1的节点层数,满足一个概率分布
    • 节点层数恰好等于1的概率为 1 − p 1-p 1p
    • 节点层数大于等于2的概率为 p p p,而节点层数恰好等于2的概率为 p ∗ ( 1 − p ) p*(1-p) p(1p)
    • 节点层数大于等于3的概率为 p 2 p^2 p2,而节点层数恰好等于3的概率为 p 2 ∗ ( 1 − p ) p^2*(1-p) p2(1p)
    • 节点层数大于等于4的概率为 p 3 p^3 p3,而节点层数恰好等于4的概率为 p 3 ∗ ( 1 − p ) p^3*(1-p) p3(1p)
  • 因此,一个节点的平均层数(即包含的平均指针数目),计算如下:

    • p = 1 / 2 p=1/2 p=1/2时,每个节点所包含的平均指针数目为2
    • p = 1 / 4 p=1/4 p=1/4时,每个节点所包含的平均指针数目为1.33
      请添加图片描述
  • SkipList的平均时间复杂度为: O ( l o g N ) O(logN) O(logN)


3.SkipList实现

  • 插入结点的关键是找到这个位置的每一层前一个结点
    • 比它小,向下走
    • 比它大,向右走
struct SkipListNode
{int _val;vector<SkipListNode*> _nextV;SkipListNode(int val, int level): _val(val), _nextV(level, nullptr){}
};class Skiplist
{typedef SkipListNode Node;
public:Skiplist(){srand(time(nullptr));_head = new Node(-1, 1); // 头节点,层数是1}bool Search(int target){Node* cur = _head;int level = _head->_nextV.size() - 1;while (level >= 0){if (cur->_nextV[level] && target > cur->_nextV[level]->_val){// 目标值比下一个结点值大,向右走cur = cur->_nextV[level];}else if (!cur->_nextV[level] || target < cur->_nextV[level]->_val){// 下一个结点是空(尾) || 目标值比下一个节点值要小,向下走level--;}else{return true;}}return false;}void Add(int num){vector<Node*> preV = FindPrevNode(num);int n = RandomLevel();Node* newnode = new Node(num, n);// 如果n超过当前最大的层数,那就升高一下_head的层数if (n > _head->_nextV.size()){_head->_nextV.resize(n, nullptr);preV.resize(n, _head);}// 链接前后结点for (size_t i = 0; i < n; i++){newnode->_nextV[i] = preV[i]->_nextV[i];preV[i]->_nextV[i] = newnode;}}bool Erase(int num){vector<Node*> preV = FindPrevNode(num);// 第一层下一个不是val,则val不在表中if (!preV[0]->_nextV[0] || preV[0]->_nextV[0]->_val != num){return false;}Node* del = preV[0]->_nextV[0];// del结点每一层前后指针链接起来for (size_t i = 0; i < del->_nextV.size(); i++){preV[i]->_nextV[i] = del->_nextV[i];}delete del;// 如果删除最高层结点,把头节点的层数也降一下// 可以稍微提高查找效率int i = _head->_nextV.size() - 1;while (i >= 0){if (!_head->_nextV[i]){i--;}else{break;}}_head->_nextV.resize(i + 1);return true;}// SkipList精髓vector<Node*> FindPrevNode(int num){Node* cur = _head;int level = _head->_nextV.size() - 1;// 插入位置每一层前一个结点指针vector<Node*> preV(level + 1, _head);while (level >= 0){if (cur->_nextV[level] && num > cur->_nextV[level]->_val){// 目标值比下一个结点值大,向右走cur = cur->_nextV[level];}else if (!cur->_nextV[level] || num <= cur->_nextV[level]->_val){preV[level--] = cur;}}return preV;}// v1.0 Cint RandomLevel(){size_t level = 1;// rand() / RAND_MAX -> [0, 1]while (rand() <= RAND_MAX * _p && level <= _maxLevel){level++;}return level;}// v2.0 C++// int RandomLevel()// {//     static std::default_random_engine generator(std::chrono::system_clock::now().time_since_epoch().count());// 	static std::uniform_real_distribution<double> distribution(0.0, 1.0);// 	size_t level = 1;// 	while (distribution(generator) <= _p && level < _maxLevel)// 	{// 		++level;// 	}// 	return level;// }
private:Node* _head;size_t _maxLevel = 32;double _p = 0.5;
};

4.SkipList VS 平衡搜索树 && Hash

  • SkipList相比平衡搜索树(AVL树和红黑树),都可以做到遍历数据有序,时间复杂度也差不多
    • SkipList的优势:
      • SkipList实现简单,容易控制
        • 平衡树增删查改遍历都更复杂
      • SkipList的额外空间消耗更低
        • 平衡树节点存储每个值有三叉链,平衡因子/颜色等消耗
        • SkipList中 p = 1 / 2 p=1/2 p=1/2时,每个节点所包含的平均指针数目为2
        • SkipList中 p = 1 / 4 p=1/4 p=1/4时,每个节点所包含的平均指针数目为1.33
  • SkipList相比哈希表而言,就没有那么大的优势了
    • SkipList劣势:
      • 哈希表平均时间复杂度是 O(1),比SkipList快
      • 哈希表空间消耗略多一点
    • SkipList优势:
      • 遍历数据有序
      • SkipList空间消耗略小一点,哈希表存在链接指针和表空间消耗
      • 哈希表扩容有性能损耗
      • 哈希表在极端场景下哈希冲突高,效率下降厉害,需要红黑树补足接力

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

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

相关文章

Linux测试服务器端口是否打开

前言 服务器端口在计算机网络通信中扮演着至关重要的角色&#xff0c;其作用可以归纳如下&#xff1a; 区分不同的应用程序或服务&#xff1a; 服务器端口用于标识和定位不同应用程序或服务在服务器上的通信入口。 通过不同的端口号&#xff0c;服务器可以同时运行多个应用程…

73. UE5 RPG 优化投射物以及敌人生成

解决发射物会与地面产生交互的问题 之前一直遇到发射物的体积过大会在发射时&#xff0c;和地面产生交互&#xff0c;我们可以调整小一些&#xff0c;然后为了防止它和自身产生交互事件。我们可以实现它在生成后&#xff0c;不会触发相关事件&#xff0c;而是在一定时间后。 对…

【STM32--Cortex-M3】

STM32-Cortex-M3 ■ Cortex-M3 处理器内核到基于Cortex-M3的MCU■ ARM的各种架构版本■ 指令集■ Cortex-M3简介■ Cortex-M3寄存器组■ Cortex-M3■ Cortex-M3■ Cortex-M3 ■ Cortex-M3 处理器内核到基于Cortex-M3的MCU Cortex-M3处理器内核是单片机的中央处理单元&#xff…

WordPress简单好看的线报主题模板源码

安装说明 到WordPress管理后台中的「外观」-「主题」中点击「添加」&#xff0c;选择baolog的主题包进行上传安装并启用即可。 提示&#xff1a;为了防止主题不兼容&#xff0c;请在安装主题前进行数据备份&#xff0c;防止数据字段重复覆盖等情况发生。 源码截图 源码下载 …

三种方式实现人车流统计(yolov5+opencv+deepsort+bytetrack+iou)

一、运行环境 1、项目运行环境如下 2、CPU配置 3、GPU配置 如果没有GPU yolov5目标检测时间会比较久 二、编程语言与使用库版本 项目编程语言使用c++,使用的第三方库,onnxruntime-linux-x64-1.12.1,opencv-4.6.0 opencv 官方地址Releases - OpenCV opencv github地址ht…

http发展史(http0.9、http1.0、http1.1、http/2、http/3)详解

文章目录 HTTP/0.9HTTP/1.0HTTP/1.1队头阻塞&#xff08;Head-of-Line Blocking&#xff09;1. TCP 层的队头阻塞2. HTTP/1.1 的队头阻塞 HTTP/2HTTP/3 HTTP/0.9 发布时间&#xff1a;1991年 特点&#xff1a; 只支持 GET 方法没有 HTTP 头部响应中只有 HTML 内容&#xff0…

七、yolov8图像标注和模型训练(目标检测)

环境配置方法&#xff1a;点这里 环境配置完毕后&#xff0c;需要进行标注工作和训练任务&#xff0c;以下分两个部分进行。 图片标注 1、按照以下的格式&#xff0c;将图片放入images中。&#xff08;不限制文件夹路径&#xff09; 2、然后下载labelme标注工具&#xff0…

循环赛日程表

描述 n 2 ^ k个选手 每个选手必须与其他n-1个选手各赛一次每个选手一天赛一次比赛打n-1天 思路 k 3时的解 我们先进行假设&#xff1a;每个选手第一天和自己比&#xff0c;然后分解成4个子问题&#xff1a; (1)14号的第14天&#xff0c;对手1~4号; (2)14号的第58天&a…

VS编译器字体颜色设置

默认颜色不好看&#xff0c;颜色之间代码各个关系之间没有很强关联性所以要设置字体颜色 颜色一步到位版本&#xff1a; 第一步&#xff1a; 第二步&#xff1a; 第三步&#xff1a;One dark Pro 第四步&#xff1a; 等待安装完后重启VS 点击Modify&#xff0c;一段时间结束后选…

IDEA 学习之 打开一个 MAVEN 工程

目录 1. 单体工程2. 多 module 工程3. 多个多 module 工程3.1. 重复 1 步骤3.2. 添加其他多 module 工程 1. 单体工程 2. 多 module 工程 3. 多个多 module 工程 3.1. 重复 1 步骤 3.2. 添加其他多 module 工程

django学习入门系列之第三点《CSS基础样式介绍2》

文章目录 文字对齐方式外边距内边距往期回顾 文字对齐方式 水平对齐方式 text-align: center;垂直对齐方式 /* 注意&#xff0c;这个只能是一行来居中 */ line-height:/*长度*/ ;样例 <!DOCTYPE html> <html lang"en"> <head><meta charset…

docker 环境部署

1.Redis部署 用docker拉取redis镜像 docker pull redis 用docker查看拉取的镜像版本号&#xff0c;这里查到的是 6.2.6 版本 docker inspect redis 通过wget指令下载对应版本的tar包&#xff0c;下载完成后解压 wget https://download.redis.io/releases/redis-6.2.6.tar.gz …

集合注意事项

目录 我们为什么要用到集合中的迭代器 List实现类的循环遍历 Set集合 HashSet TreeSet Map Hashmap Treemap Hashtable map的遍历方式 Collections的一些静态方法 我们为什么要用到集合中的迭代器 List实现类的循环遍历 如图我们对arraylist中加入了三个相同的“a”…

多路h265监控录放开发-(8)通过XCameraWIget类拖拽实现指定播放rtsp和窗口

首先修改xviewer的构造函数&#xff0c;把创建QWiget对象改为XCameraWiget对象&#xff0c;执行XCameraWiget类的构造函数 xcamera_widget.h #pragma once #include <QWidget> class XCameraWidget :public QWidget {Q_OBJECTpublic:XCameraWidget(QWidget* p nullptr)…

网络安全:Web 安全 面试题.(SQL注入)

网络安全&#xff1a;Web 安全 面试题.&#xff08;SQL注入&#xff09; 网络安全面试是指在招聘过程中,面试官会针对应聘者的网络安全相关知识和技能进行评估和考察。这种面试通常包括以下几个方面&#xff1a; &#xff08;1&#xff09;基础知识:包括网络基础知识、操作系…

容器之事件盒

代码&#xff1a; #include <gtk-2.0/gtk/gtk.h> #include <glib-2.0/glib.h> #include <gtk-2.0/gdk/gdkkeysyms.h> #include <stdio.h>static void label_const(GtkWidget *eventbox) {static int i 0;static char citem[100];sprintf(citem, &quo…

【Unity学习笔记】第十八 基于物理引擎的日月地系统简单实现

转载请注明出处: https://blog.csdn.net/weixin_44013533/article/details/139701843 作者&#xff1a;CSDN|Ringleader| 目录 目标数学理论资源准备数据准备代码实现Unity准备效果展示注意事项后记 目标 目标&#xff1a;利用Unity的物理引擎实现 “日地月三体系统” 。 效果…

PHP转Go系列 | ThinkPHP与Gin的使用姿势

大家好&#xff0c;我是码农先森。 安装 使用 composer 进行项目的创建。 composer create-project topthink/think thinkphp_demo使用 go mod 初始化项目。 go mod init gin_demo目录 thinkphp_demo 项目目录结构。 thinkphp_demo ├── LICENSE.txt ├── README.md …

【干货】微信小程序免费开源项目合集

前言 2024年了&#xff0c;还有小伙伴在问微信小程序要怎么开发&#xff0c;有什么好的推荐学习项目可以参考的。今天分享一个收集了一系列在微信小程序开发中有用的工具、库、插件和资源&#xff1a;awesome-github-wechat-weapp。 开源项目介绍 它提供了丰富的资源列表&…

day01-Numpy的安装

numpy的安装 同样&#xff0c;anaconda内置有Numpy包 Numpy是用c语言实现的&#xff0c;运算速度比python快得多 import numpy as np np.__version__out: 1.18.5使用Jupyter编辑器打印numpy包的版本 NumPy ndarray对象 NumPy定义了一个n维数组对象&#xff0c;简称ndarra…