【STL源码剖析】deque 的使用

别院深深夏席清,石榴开遍透帘明。

树阴满地日当午,梦觉流莺时一声。


目录

deque 的结构

deque 的迭代器剖析

deque 的使用

​编辑

deque 的初始化

deque 的容量操作

deque 的访问操作 

在 pos 位置插入另一个向量的 [forst,last] 间的数据​编辑

删除 [first,last] 之间的元素

assign的用法

deque 的优缺点

 契子


deque ( double-ended queue ,双端队列是有下标顺序容器,它允许在其首尾两段快速插入及删除。另外,在 deque 任一端插入或删除不会非法化指向其余元素的指针或引用。 

deque 是一块连续的空间(至少逻辑看来如此),连续空间我们可能会想到 array(数组)或者vectorarray 无法成长,vector 虽然可以成长但是只有尾端成长,而且扩容(成长)只是假象,实际上是:<1>另寻更大的空间、<2>将原资料拷贝过去、<3>释放原空间。这样的话 vector 成长所带来的代价是相当高的。

deque 的想象结构:就是相同的连续空间拼接而成 

deque 巧妙的避开了 vector 的缺点,deque 的结构有一段一段的定量空间组成。一旦有必要在 deque 的前端或者尾端增加新空间,便配置一定量的连续空间,串在整个 deque 的头端或者尾端。deque 便是在这些分段的定量连续空间上,维序其连续的假象,并提供随机存取的界面。简单来讲就是 vectorlist 的结合体,大小相同的连续空间串在一起。这样就避开了 vector 的【重新配置、拷贝、释放】的轮回。不过呢?有舍就有得,代价是迭代器的框架很复杂。

现在我们已经对 deque 有了一定的了解 ~ 接下来我们看看它的结构


deque 的结构

vector 容器采用连续的线性空间不同,deque 容器存储数据的空间是由一段一段等长的连续空间构成,各段空间之间并不一定是连续的,可以位于在内存的不同区域。为了管理这些连续空间,deque 容器用数组 map (注意,不是 STL 的 map)存储着各个连续空间的首地址。也就是说,map 数组中存储的都是指针,指向那些真正用来存储数据的各个连续空间。我们称 map 数组为中控区,存储的空间位置为缓冲区

这个时候我们的老铁可能就会问, 要是中控区的空间满了怎么办,我们知道 deque 是一块连续空间拼接而成的,一般情况下空间是足够的,并不会频繁扩容。如果 map 不够用了,我们这边需要找一块更大的空间来做我们的中控器 map,这个时候 map 就会有更多的节点空间来存储我们的缓冲区。

 我们再来细致的了解一下中控器、缓冲区、迭代器之间的相互关系:

简单来讲我们的迭代器就是一块 4 个空间的节点,cur 指向缓冲区的数据元素,first 指向缓冲区的首地址,last 指向缓冲区的末地址,而 node 则指向在中控区的节点,那如果我们要访问 deque 内的数据该怎么办呢 ?

举个栗子 ~ 我想访问 12 位置的元素 

首先我们先来分析一下:

这是一个空间大小为 8 的数组 Buff(通常用 Buff 来表示缓冲区的空间)

我们要找到是第几个 Buff 数组存放了 12 位置:i = pos/N(设 pos 是访问位置,N 是 Buff 空间的大小)所以 i = 12/8,即第二个 Buff 数组存放了该访问元素

然后我们要找到该元素在 Buff 空间的第几号位置:j = pos%N,j = 12%8,即该元素在 Buff 空间的第 4 号位置(Buff 空间从 0 开始)


deque 的迭代器剖析

deque 容器底层将序列中的元素分别存储到了不同段的连续空间中,因此要想实现迭代器的功能,必须先解决如下 2 个问题:

迭代器在遍历 deque 容器时,必须能够确认各个连续空间在 map 数组中的位置
迭代器在遍历某个具体的连续空间时,必须能够判断自己是否已经处于空间的边缘位置。如果是,则一旦前进或者后退,就需要跳跃到上一个或者下一个连续空间中

为了控制 deque 的迭代功能,deque 专门设置了以下的结构:

template <class T ,...,size_t BufSiz>
struct __deque_iterator 
{ // ...typedef T** map_pointer;
private:T* cur; T* first; T* last; map_pointer node;
};
cur指向迭代器当前在缓冲区遍历的元素
first指向缓冲区的首节点
last指向缓冲区的末节点
node指向中控区的节点

借助这 4 个指针,deque 迭代器对各种运算符进行了重载、封装:

template <class T,..., size_t BufSiz>
struct __deque_iterator
{//...typedef __deque_iterator self;typedef T** map_pointer;typedef ptrdiff_t difference_type;
public:inline size_t __deque_buf_size(size_t n, size_t sz){return n != 0 ? n : (sz < 512 ? size_t(512 / sz) : size_t(1));}static size_t buffer_size(){return __deque_buf_size(0, sizeof(T)); }void set_node(map_pointer new_node){//记录新的空间在 map 中的位置node = new_node;first = *new_node; //更新 last 指针,difference_type(buffer_size())表示每段连续空间的长度last = first + difference_type(buffer_size());}T* operator*() const {return *cur;}T* operator->() const{return &(operator *()); }self& operator++() {++cur;if (cur == last) {//如果 cur 位于连续空间边缘,则先将迭代器跳跃到下一个连续空间中set_node(node + 1);cur = first;}return *this;}self& operator--(){if (cur == first) {//如果 cur 位于连续空间边缘,则先将迭代器跳跃到前一个连续空间中set_node(node - 1);cur == last;}--cur;return *this;}T* operator[](difference_type n) const { return *(*this + n);}bool operator==(const self& x) const { return cur == x.cur;}bool operator!=(const self& x) const { return !(*this == x); }private:T* cur;T* first;T* last;map_pointer node;
};

 

关于我们 typedef ptrdiff_t 具体是什么:不同机器下,他都有对应的类是为了适应不同平台下定义的

我们的迭代器就已经大致完成了 ~

这样就可以借助 start finish,以及 deque 迭代器中重载的诸多运算符,就可以实现 deque 容器提供的大部分成员函数:

template <class T, size_t BufSiz = 0>
class deque 
{typedef __deque_iterator<T, ..., BufSiz> iterator;typedef pointer* map_pointer;
public: iterator begin(){ return start; }iterator end(){ return finish;}T* operator[](size_type n) {return start[difference_type(n)];}T* front() {return *start;}T* back() {iterator tmp = finish;--tmp;return *tmp;}size_t size() const { return finish - start;}bool empty() const { return finish == start; }
protected:iterator start; iterator finish; map_pointer map; size_t map_size;
};

对了以上的代码都是伪代码,不能使用的哦 ~ 只是为了更好了解 deque 的结构 

需要源码的老铁可以点击文章顶部的资源管理

接下来我们聊一聊使用:

deque 的使用

在讲使用之前,我们可以先来参考一下文档 deque 文档

deque 的初始化

deque<int> a;        // 定义一个int类型的双端队列a
deque<int> a(10);    // 定义一个int类型的双端队列a,并设置初始大小为10
deque<int> a(10, 1); // 定义一个int类型的双端队列a,并设置初始大小为10且初始值都为1
deque<int> b(a);     // 定义并用双端队列a初始化双端队列b
deque<int> b(a.begin(), a.begin()+3); // 将双端队列a中从第0个到第2个(共3个)作为双端队列b的初始值
deque<int> b=a;      // 定义一个int类型的双端队列b,并将双端队列a赋值给b

我们之前学过 vectorlist 所以应该很容易理解以上的意义:

无参构造、链式构造、 拷贝构造、迭代器区间构造、赋值构造

这里就不一一讲解了

注意:除此之外,deque 还可以直接使用数组来初始化

	int a[] = { 1,2,3,4,5 };deque<int> str(a, a+5);

我们简单来测试一下 ~  

void deque_test()
{int a[] = { 1,2,3,4,5 };deque<int> str(a, a + 5);while (!str.empty()){cout << str.front() << " ";str.pop_front();}
}

deque 的容量操作

size() 计算容器大小
max_size() 计算容器最大容量 (不经常用的接口函数)
resize() 预留容器的空间大小
empey() 判断容器是否为空

deque 的访问操作 

<1> 支持下标 [ ] 访问:越界报错

<2> 支持 at 访问:越界抛异常

<3> front:访问第一个元素

<4> back:访问最后一个元素

代码测试:

void deque_test()
{deque<int> str = { 1,2,3,4,5 };cout << str.front() << endl;cout << str.back() << endl;
}

 

老铁们 ~ 我想开摆了(挑重点讲):

在 pos 位置插入另一个向量的 [forst,last] 间的数据

注意:插入元素不一定要同一个容器,但是要同一种类型 

代码测试:

void deque_test()
{deque<int> str = { 1,2,3,4,5 };vector<int> arr = { 7,8,9,10,11 };str.insert(str.begin()+2, arr.begin(), arr.end());while (!str.empty()){cout << str.front() << " ";str.pop_front();}
}


 

删除 [first,last] 之间的元素

代码测试: 

void deque_test()
{deque<int> str = { 1,2,3,4,5 };str.erase(str.begin(), str.begin()+3);while (!str.empty()){cout << str.front() << " ";str.pop_front();}
}


assign的用法

assign 简单来讲就是初始化,而且不仅支持迭代器区间初始化还支持用 n val 初始化

代码测试 

void deque_test()
{deque<int> str;str.assign(5, 0);deque<int> arr;arr.assign(str.begin(), str.end());while (!arr.empty()){cout << arr.front() << " ";arr.pop_front();}
}


 

deque 的优缺点

我们先来看看 vectorlist 的特点:

 

deque 的优势:

<1> 与 vector 比较:头部插入和删除时,不需要搬移元素,效率特别高,而且在扩容时,也不需要搬移大量的元素,因此其效率是比 vector 高的

<2> 与 list 比较:其底层是连续空间,空间利用率比较高,不需要存储额外字段

deque的缺点:

deque 不适合遍历,因为在遍历时,deque 的迭代器要频繁的去检测其是否移动到某段小空间的边界,导致效率低下,而排序场景中,可能需要经常遍历,因此在实际中,需要线性结构时,大多数情况下优先考虑 vector listdeque 的应用并不多,不适合大量的头部和中间插入删除,也不适合大量的随机访问。而目前能看到的一个应用就是,STL 用其作为 stack queue 的底层数据结构。

使用场景: 

(1)如果你需要高效的随即存取,而不在乎插入和删除的效率,使用 vector
(2)如果你需要大量的插入和删除,而不关心随机存取,则应使用 list
(3)如果你需要随机存取,而且关心两端数据的插入和删除,则应使用 deque

我们上次说起 stackqueue 空间适配器都有 duque 的影子,那么为什么 deque 会作为 stack queue 的底层结构呢?

stack queue 不需要遍历(因此 stack queue 没有迭代器),只需要在固定的一端或者两端进行操作
stack 中元素增长时,dequevector 的效率高(扩容时不需要搬移大量数据),queue 中的元素增长时,使用 deque 作为底层默认容器,不仅效率高,而且内存使用率高

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

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

相关文章

【人工智能Ⅱ】实验9:强化学习Q-Learning算法

实验9&#xff1a;强化学习Q-Learning算法 一&#xff1a;实验目的 1&#xff1a;了解强化学习的基本概念。 2&#xff1a;学习强化学习经典算法——Q-Learing算法。 3&#xff1a;通过Q-Learing算法解决问题。 二&#xff1a;实验内容 2.1 强化学习 强化学习&#xff08;…

iOS18新功能大爆料,打破常规,全面升级,这些变化不容错过!

众所周知&#xff0c;苹果 iOS 操作系统近年来都没有发生重大变化&#xff0c;主要是添加小部件、锁屏编辑和手机屏幕编辑等功能&#xff0c;再加上bug偏多&#xff0c;以至于越来越多iPhone用户不愿意再升级系统了。这一点&#xff0c;从 iOS 17 明显降低的安装率中就能看出一…

对人脸图像进行性别和年龄的判断

判断性别和年龄 导入必要的库加载预训练的人脸检测模型加载预训练的性别和年龄识别模型定义性别和年龄的标签列表创建Tkinter窗口&#xff1a;定义选择图片的函数&#xff1a;创建一个按钮&#xff0c;用于打开文件选择对话框定义显示图片的函数创建预测性别和年龄的函数创建预…

Docker大学生看了都会系列(二、Mac通过Homebrew安装Docker)

系列文章目录 第一章 Docker介绍 第二章 Mac通过Homebrew安装Docker 文章目录 前言Mac通过Homebrew安装本机环境系统要求terminal命令安装查看安装信息配置阿里云镜像加速登陆阿里云配置加速地址其他国内加速地址 总结 前言 在上一章了解了Docker容器是什么之后&#xff0c;本…

solidworks二维样条曲线使用实例

单位mm 绘制一个圆 直径为50mm&#xff0c; 基准面 上视基准面&#xff0c;距离50mm&#xff0c; 2个六边形 一个内嵌圆 另一个直径60mm&#xff0c; 将两个六边形改成构造线 选择样条曲线&#xff0c;将六边形的顶点连接在一起 放样曲面 插入–曲面–放样曲面 平面区域…

RabbitMQ不完整的笔记

同步的不足 1、拓展性差&#xff0c;当要添加功能时&#xff0c;需要在原来的功能代码上做修改&#xff0c;高耦合。 2、性能下降&#xff0c;调用者需要等待服务提供者执行完返回结果后&#xff0c;才能继续向下执行 3、级联失败&#xff0c;由于我们是基于OpenFeign调用交易…

Visual Studio Code使用(C++项目新建,运行)

VS Code 直接在官网下载安装。 接下来安装插件&#xff0c;下图是C所需的对应插件 1.新建项目 VS Code下载安装完成后&#xff0c;直接进入欢迎页&#xff1a; 在访达/文件夹中新建一个文件夹&#xff0c;欢迎页点击【打开】&#xff0c;选择刚刚新建的文件夹。点击第一个图…

华为OD机试 - 最大坐标值(Java 2024 D卷 100分)

华为OD机试 2024C卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;A卷B卷C卷&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;每一题都有详细的答题思路、详细的代码注释、样例测试…

数据结构第三篇【链表的相关知识点一及在线OJ习题】

数据结构第三篇【链表的相关知识点一及在线OJ习题】 链表链表的实现链表OJ习题顺序表和链表的区别和联系 本文章主要讲解关于链表的相关知识&#xff0c;喜欢的可以三连喔 &#x1f600;&#x1f603;&#x1f604;&#x1f604;&#x1f60a;&#x1f60a;&#x1f643;&#…

Makefile的入门学习

一、Makefile的入门学习 编译工具及构建工具介绍 在之前的课程&#xff0c;都是直接使用gcc对代码进行编译&#xff0c;这对简单的工程是可以的&#xff0c;但当我们遇到复杂的工程时&#xff0c;每次用gcc等编译工具去操作就会显得很低效。因此make工具就出现了&#xff0c;…

LeetCode1137第N个泰波那契数

题目描述 泰波那契序列 Tn 定义如下&#xff1a; T0 0, T1 1, T2 1, 且在 n > 0 的条件下 Tn3 Tn Tn1 Tn2给你整数 n&#xff0c;请返回第 n 个泰波那契数 Tn 的值。 解析 递归应该会超时&#xff0c;可以用循环&#xff0c;或者官方解法的矩阵的幂。 public int tr…

(4) qml动态元素

文章目录 概述注意 动画元素变化的策略Animation on 变化behavior on⽤standalone animation注意 缓冲曲线&#xff08;Easing Curves&#xff09;动画分组 概述 这⼀章介绍如何控制属性值的变化&#xff0c;通过动画的⽅式在⼀段时间内来改变属性值。这项技术是建⽴⼀个现代化…

获取(复制)网页上的文字

获取&#xff08;复制&#xff09;网页上的文字 今天在搜索历史课本上一段文言文的翻译时&#xff0c;找到的网页&#xff0c;屏蔽了右键&#xff0c;不能选择&#xff0c;当然不让复制啦。对于这样的网站可以采用如下方法进行数据的获取&#xff0c;以chrome为例。 1、网页另…

keil5常见使用技巧记录(更新)

快速到函数定义 F12或自己定义快捷键CTRLK&#xff08;个人设定&#xff09; 修改快捷键 下图实例是快速跳转到函数或变量定义位置&#xff0c;当然可以定义其他功能快捷键&#xff0c;如快速注释多行&#xff0c;快速消除注释等 标记全部查找变量的蓝色框取消 CTRLshiftF2…

【YOLOv10改进[Backbone]】图像修复网络AirNet助力YOLOv10目标检测效果 + 含全部代码和详细修改方式 + 手撕结构图 + 全网首发

本文带来的是图像复原网络AirNet&#xff0c;它由基于对比度的退化编码器( CBDE )和退化引导的恢复网络( DGRN )两个模块组成。可以在一个网络中恢复各种退化图像。AirNet不受损坏类型和级别的先验限制&#xff0c;仅使用观察到的损坏图像进行推理。本文中将使用图像修复网络Ai…

使用Python绘制瀑布图

使用Python绘制瀑布图 瀑布图效果代码 瀑布图 瀑布图&#xff08;Waterfall Chart&#xff09;是一种数据可视化工具&#xff0c;用于展示累积数值的变化&#xff0c;尤其适合于展示随时间或过程中的增减变化。它通常用于财务分析&#xff0c;如展示收入、支出和净利润的变化过…

【离散数学】数理逻辑集合论知识点汇总

期末题型&#xff1a; 一、 单选题&#xff08;每题2分&#xff0c;10题共20分&#xff09; 命题判定、哈斯图边计算等 二、 填空题&#xff08;每空1分&#xff0c;共20分&#xff09; 与非和或非的表示等 三、 简答题&#xff08;10题&#xff0c;每题6分&#xff0c;共60分&…

安装禅道,帮助测试,测试打磨项目精度。

先检查docker版本。 sudo docker network create --subnet172.172.172.0/24 zentaonet sudo docker run --name zentao2 -p 8080:80 -p 3307:3306 --networkzentaonet --ip 172.172.172.3 -e MYSQL_INTERNALtrue -v /media/cykj/3T/ze…

【十年java搬砖路】Jumpserver docker版安装及配置Ldap登陆认证

Jumpserver docker 安装启动教程 拉取镜像 docker pull JumpServer启动进行前确保有Redis 和Mysql 创建jumperServer数据库 在MYSQL上执行 创建数据库 登陆MYSQL mysql -u root -p 创建Jumperserveri库 create database jumpserver default charset utf8mb4;可以为jumperSe…

【软件开发】Java学习路线

本路径视频教程均来自尚硅谷B站视频&#xff0c;Java学习课程我已经收藏在一个文件夹下&#xff0c;B站文件夹同时会收藏其他Java视频&#xff0c;感谢关注。指路&#xff1a;https://www.bilibili.com/medialist/detail/ml3113981545 2024Java学习路线&#xff08;快速版&…