【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.mfbz.cn/a/676826.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

JVMの堆、栈内存存储

1、JVM栈的数据存储 通过前面的学习&#xff0c;我们知道&#xff0c;将源代码编译成字节码文件后&#xff0c;JVM会对其中的字节码指令解释执行&#xff0c;在解释执行的过程中&#xff0c;又利用到了栈区的操作数栈和局部变量表两部分。 而局部变量表又分为一个个的槽位&…

web安全基础学习笔记

这里写目录标题 1.使用hackbar2.php漏洞基本分析 弱类型语言2.2 php漏洞找到隐藏的源代码之 index.php~2.3 php漏洞找到隐藏的源代码之 vim的临时文件 /.index.php.swp3.php漏洞基本分析 数组 3.php漏洞基本分析 extract4.php漏洞基本分析 strpos eregi函数漏洞4.php漏洞基本分…

Java web应用性能分析之【java进程问题分析定位】

Java web应用性能分析之【java进程问题分析概叙】-CSDN博客 Java web应用性能分析之【java进程问题分析工具】-CSDN博客 Java web应用性能分析之【jvisualvm远程连接云服务器】-CSDN博客 由于篇幅限制、前面三篇讲了准备工作和分析小结&#xff0c;这里将详细操作java进程问题…

The First项目报告:去中心化知识产权治理协议MON Protocol如何革新链游产业?

2023年12月&#xff0c;RPG NFT 游戏 Pixelmon 首席执行官 GiulioX 在 X 平台表示&#xff0c;确认将推出代币 MON&#xff0c;代币生成&#xff08;TGE&#xff09;时间将取决于 MON 的路线图和主流 CEX 的启动板队列。12 月 11 日&#xff0c;RPG NFT 游戏 Pixelmon 首席执行…

防爆AGV叉车在现代物流行业的应用

AGV 随着机器人技术在中国的快速发展&#xff0c;国内企业开始推出区别于传统叉车的叉车AGV&#xff0c;旨在为企业降本增效&#xff0c;降低人工成本与对人的依赖&#xff1b;同时&#xff0c;也将人工从危险恶劣的环境中解放出来。随着技术的持续提升&#xff0c;叉车AGV已经…

API低代码平台介绍4-数据库记录插入功能

数据库记录插入功能 本篇文章我们将介绍如何使用ADI平台定义一个向目标数据库插入记录的接口&#xff0c;包括手工组装报文单表插入、手工组装报文多表插入、自动组装报文多表插入三种方式。无论是单表插入还是多表插入&#xff0c;任何一条记录写入失败&#xff0c;那么默认情…

kvm学习 - 迅速上手示例

目录 kvmtool kvmsample kvmtool GitHub - kvmtool/kvmtool: Stand-alone Native Linux KVM Tool repoStand-alone Native Linux KVM Tool repo. Contribute to kvmtool/kvmtool development by creating an account on GitHub.https://github.com/kvmtool/kvmtool.git cd …

17.Redis之主从复制

1.主从复制是怎么回事&#xff1f; 分布式系统, 涉及到一个非常关键的问题: 单点问题 单点问题&#xff1a;如果某个服务器程序, 只有一个节点(只搞一个物理服务器, 来部署这个服务器程序) 1.可用性问题,如果这个机器挂了,意味着服务就中断了~ 2.性能/支持的并发量也是比较有限…

C语言学习:数据类型

一、 为什么要引入数据类型 • 计算机中每个字节都有一个地址&#xff08;类似门牌号&#xff09; • CPU通过 地址 来访问这个字节的空间 0x20001103 1 0 0 1 0 0 1 1 0x20001102 1 1 1 0 1 1 1 0 0x20001101 1 1 1 1 0 1 0 1 0x20001100 0 …

【UML用户指南】-06-面向对象建模-关系(relationship)

目录 1、面向对象建模常见的关系 2、关系的组成元素 3、依赖关系 4、泛化关系 5、关联关系 关联的四种修饰 1.名称 2.角色 3.多重性 4.聚合 6、常用建模技术 6.1、对简单依赖建模 6.2、对单继承建模 6.3、对结构关系建模 1、面向对象建模常见的关系 依赖 &#x…

flask轻松入门,概念讲解

Hello World Flask 是轻量级web框架&#xff0c;仅保留了核心功能&#xff1a; 请求响应处理模板渲染URL路由 文章目录 Hello Worldflask命令模式python命令模式两种模式对比修改入口文件配置flask命令修改python命令修改 修改端口和地址flask命令修改python命令修改 修改 URL …

jupyter之plt 画图弹出窗口展示图片以及静态图片切换方法

1. jupyter出图的三种方式 在python的Jupyter Notebook中&#xff0c;使用matplotlib绘制动态图形时&#xff0c;可能出现只显示一张静态图像。 这是因为在notebook中使用plt绘图共有三种模式&#xff1a; %matplotlib inline&#xff1a;这是默认的模式&#xff0c;输出的图片…

JavaScript 基础 - 对象

对象 对象是一种无序的数据集合&#xff0c;可以详细的描述描述某个事物。 注意数组是有序的数据集合。它由属性和方法两部分构成。 语法 声明一个对象类型的变量与之前声明一个数值或字符串类型的变量没有本质上的区别。 <script>let 对象名 {属性名&#xff1a;属性值…

【OPENMV】学习记录 (持续更新)

一、图像 1 设置彩色&#xff0f;黑白&#xff1a; sensor.set_pixformat() 设置像素模式。 sensor.GRAYSCALE: 灰度&#xff0c;每个像素8bit。sensor.RGB565: 彩色&#xff0c;每个像素16bit。 2 设置图像大小&#xff1a; sensor.set_framesize() 设置图像的大小 sensor.…

前端优化之图片压缩——tinyPNG

今天前端前辈新介绍的一个压缩图片的工具——tinyPNG&#xff0c;地址&#xff1a;TinyPNG – Compress WebP, PNG and JPEG images intelligently可以将图片压缩&#xff0c;进行优化。 一、使用方法——手动压缩 将超过200kb的图片拖到我标注的红框框里&#xff0c;拖到这里…

如何快速定位到影响mysql cpu飙升的原因——筑梦之路

通常我们只需要执行show processlist 进行查看&#xff0c;一般执行时间最长的SQL八九不离十就是罪魁祸首&#xff0c;但当show processlist的输出有近千条&#xff0c;那么很难第一眼就发现有问题的SQL&#xff0c;那么如何快速找到呢&#xff1f;其实也非常简单。我们知道mys…

多卡聚合智能融合通信设备在无人机无线视频传输应用

无人驾驶飞机简称“无人机”&#xff0c;是利用(无线电)遥控设备和自备的程序控制装置操纵的不载人飞行器&#xff0c;现今无人机在航拍、农业、快递运输、测绘、新闻报道多个领域中都有深度的应用。 无人机无线视频传输保证地面人员利用承载的高灵敏度照相机可以进行不间断的画…

HCIA--OSPF实验(复习)

实验拓扑&#xff1a; 实验思路&#xff1a; 1.规划IP&#xff0c;配置环回&#xff0c;接口IP 2.把R1&#xff0c;R2优先级改为0&#xff0c;让R1、R2放弃选举&#xff0c; [r1]interface g0/0/0 [r1-GigabitEthernet0/0/0]ospf dr-priority 0 <r1>reset ospf…

【linux】线程同步和生产消费者模型

线程同步 当我们多线程访问同一个临界资源时&#xff0c;会造成并发访问一个临界资源&#xff0c;使得临界资源数据不安全&#xff0c;我们引入了锁的概念&#xff0c;解决了临界资源访问不安全的情况&#xff0c;对于线程而言竞争锁的能力有强有弱&#xff0c;对于之前就抢到…

图形学初识--颜色混合

文章目录 前言正文为什么要有颜色混合&#xff1f;颜色混合常见实现方式&#xff1f;上述颜色混合注意点 结尾&#xff1a;喜欢的小伙伴点点关注赞哦! 前言 本章节补充一下颜色混合的内容&#xff0c;主要包含&#xff1a;为什么要有颜色混合&#xff1f;颜色混合常实现方式&a…