C++初阶学习——探索STL奥秘——标准库中的queue与stack

1、适配器模式

STL 中的适配器可以分为三类:

从应用角度出发

容器适配器 container adapters

迭代器适配器 iterator adapters

仿函数适配器 functor adapters

其中,容器适配器可修改底层为指定容器

如由 vector 构成的栈、由 list 构成的队列

迭代器适配器可以实现其他容器的反向选代器

最后的仿函数适配器就厉害了

几乎可以无限制的创造出各种可能的表达式

本文介绍的是容器适配器,栈和队列都用到了容器适配器

最后还会介绍一下常作为这两种容器适配器的默认底层容器双端队列

2.stack

2.1介绍

1. stack是一种容器适配器,专门用在具有后进先出操作的上下文环境中,其删除只能从容器的一端进行 元素的插入与提取操作。

2. stack是作为容器适配器被实现的,容器适配器即是对特定类封装作为其底层的容器,并提供一组特定的成员函数来访问其元素,将特定类作为其底层的,元素特定容器的尾部(即栈顶)被压入和弹出。

3.可以看出,栈有两个模板参数
参数1:T 栈中的元素类型,同时也是底层容器中的元素类型
参数2: container 实现栈时用到的底层容器,这里为缺省参数,缺省结构为双端队列 deque

4. stack的底层容器可以是任何标准的容器类模板或者一些其他特定的容器类,这些容器类应该支持以下操作:

empty:判空操作

back:获取尾部元素操作

push_back:尾部插入元素操作

pop_back:尾部删除元素操作

5. 标准容器vector、deque、list均符合这些需求,默认情况下,如果没有为stack指定特定的底层容器, 默认情况下使用deque。

2.2使用

可通过刷题来巩固对stack使用的理解

 通过一个栈来存储数据,另一个栈来存储最小值

 

class MinStack
{
public: void push(int x){ // 只要是压栈,先将元素保存到_elem中_elem.push(x);// 如果x小于_min中栈顶的元素,将x再压入_min中if(_min.empty() || x <= _min.top())_min.push(x);}void pop(){// 如果_min栈顶的元素等于出栈的元素,_min顶的元素要移除if(_min.top() == _elem.top())_min.pop();_elem.pop();}int top(){return _elem.top();}int getMin(){return _min.top();}private:// 保存栈中的元素std::stack<int> _elem;// 保存栈的最小值std::stack<int> _min;
};

 

class Solution {
public:bool IsPopOrder(vector<int> pushV,vector<int> popV) {//入栈和出栈的元素个数必须相同if(pushV.size() != popV.size())return false;// 用s来模拟入栈与出栈的过程int outIdx = 0;int inIdx = 0;stack<int> s;while(outIdx < popV.size()){// 如果s是空,或者栈顶元素与出栈的元素不相等,就入栈while(s.empty() || s.top() != popV[outIdx]){if(inIdx < pushV.size())s.push(pushV[inIdx++]);elsereturn false;}// 栈顶元素与出栈的元素相等,出栈s.pop();outIdx++;}return true;}
};

 

class Solution {
public:int evalRPN(vector<string>& tokens) {stack<int> s;for (size_t i = 0; i < tokens.size(); ++i){string& str = tokens[i];// str为数字if (!("+" == str || "-" == str || "*" == str || "/" == str)){s.push(atoi(str.c_str()));}else{// str为操作符int right = s.top();s.pop();int left = s.top();s.pop();switch (str[0]){case '+':s.push(left + right);break;case '-':s.push(left - right);break;case '*':s.push(left * right);break;case '/':// 题目说明了不存在除数为0的情况s.push(left / right);break;}}}return s.top();}
};

2.3模拟实现

#pragma once
#include <vector>using namespace std;namespace Yohifo
{//这里选择模板参数2 底层容器 的缺省值为 vectortemplate<class T, class Container = vector<int>>class stack{public:/*创建出Container的对象_c的时候就已经调用了Container中的默认构造,所以这边不必特地编写stack的默认构造函数*/stack(){}{}//不需要显式的去写析构函数,默认生成的够用了//同理拷贝构造、赋值重载也不需要bool empty() const{return _c.empty();}size_t size() const{return _c.size();}//top 需要提供两种版本T& top(){return _c.back();}const T& top() const{return _c.back();}//选取的底层容器必须支持尾部操作void push(const T& val){_c.push_back(val);}void pop(){//空栈不能弹出,可在底层容器中检查出来_c.pop_back();}private:Container _c;	//成员变量为具体的底层容器};
}

适配器的厉書之处就在于 只要底层容器有我需要的函数接口,那么我就可以为其适配出一个容器适配器,比如 vector 构成的栈、list构成的栈、 deque 构成的栈,甚至是 string 也能适配出一个栈

只要符合条件,都可以作为栈的底层容器,当然不同结构的效率不同,因此库中选用的是效率较高的 deque 作为默认底层容器

3.queue

3.1介绍

1.队列是一种容器适配器,专门用于在FIFO上下文(先进先出)中操作

其中从容器一端插入元素,另一端提取元素。

2. 队列作为容器适配器实现,容器适配器即将特定容器类封装作为其底层容器类,queue提供一组特定的成员函数来访问其元素。

元素从队尾入队列,从队头出队列。

3. 底层容器可以是标准容器类模板之一,也可以是其他专门设计的容器类。

该底层容器应至少支持以下操作:

empty:检测队列是否为空

size:返回队列中有效元素的个数

front:返回队头元素的引用

back:返回队尾元素的引用

push_back:在队列尾部入队列

pop_front:在队列头部出队列

4. 标准容器类deque和list满足了这些要求。默认情况下,如果没有为queue实例化指定容器类,则使用标准容器deque。  

3.2queue的使用

 同样我们可以通过刷题巩固对queue使用时的理解

class MyStack {
private:queue<int>q1;queue<int>q2;
public:MyStack() {}void push(int x) {//q1用来存储数据,q2仅起辅助作用q1.push(x);}int pop() {int n=q1.size()-1;//把q1前n-1个元素放入q2,取最后一个,q2再重新放入q1int ans=0;while(n--){q2.push(q1.front());q1.pop();}ans=q1.front();q1.pop();while(!q2.empty()){q1.push(q2.front());q2.pop();}return ans;}int top() {return q1.back();}bool empty() {return q1.empty();}
};/*** Your MyStack object will be instantiated and called as such:* MyStack* obj = new MyStack();* obj->push(x);* int param_2 = obj->pop();* int param_3 = obj->top();* bool param_4 = obj->empty();*/

/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode() : val(0), left(nullptr), right(nullptr) {}*     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}*     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}* };*/
class Solution {
public:vector<vector<int>> levelOrder(TreeNode* root) {vector<vector<int>> ret;if(root==nullptr){return ret;}queue<TreeNode*> q;q.push(root);while(!q.empty()){   ret.push_back(vector<int>());int levelsize=q.size();for(int i=1;i<=levelsize;i++){if(q.front()->left)q.push(q.front()->left);if(q.front()->right)q.push(q.front()->right);ret.back().push_back(q.front()->val);q.pop();}}return ret;}
};

 3.3模拟实现

原理和stack差不多

#pragma once
#include <list>using namespace std;namespace Yohifo
{template<class T, class Container = list<T>>class queue{public:queue(){}/*创建出Container的对象_c的时候就已经调用了Container中的默认构造,所以这边不必特地编写queue的默认构造函数*/                        //这里也不需要提供拷贝构造、赋值重载、析构函数bool empty() const{return _c.empty();}size_t size() const{return _c.size();}//选取的底层容器中,已经准备好了相关函数,如 front、backT& front(){return _c.front();}const T& front() const{return _c.front();}T& back(){return _c.back();}const T& back() const{return _c.back();}void push(const T& val){_c.push_back(val);	//队列只能尾插}void pop(){_c.pop_front();	//队列只能头删}private:Container _c;	//成员变量为指定的底层容器对象};
}

4.总结

栈和队列在实际开发中作为一种辅助结构被经常使用,比如内存空间划分中的栈区,设计规则符合栈FILO;

操作系统中的各种队列,如

阻塞队列,设计规则符合队列 FIFO。

除此以外,在很多OJ题中,都需要借助栈和队列进行解题

注意:
栈和队列都属于特殊的数据结构,原则上是不支持遍历的,因为一旦进行遍历,其中的数据必然被弹出,因此两者都没有提供迭代器

假设容器没有提供头尾操作,比如 map 和 set ,那么就不能拿它们适配出栈或队列,强行使用会报错

5.双端队列deque(了解就行)

5.1介绍

deque(双端队列):是一种双开口的"连续"空间的数据结构,双开口的含义是

可以在头尾两端进行插入和删除操作,且时间复杂度为O(1)

与vector比较,头插效率高,不需要搬移元素。

与list比较,空间利用率比较高。

deque并不是真正连续的空间,而是由一段段连续的小空间拼接而成的,实际deque类似于一个动态的二维数组,其底层结构如下图所示:

双端队列底层是一段假象的连续空间,实际是分段连续的

deque 的扩容机制:只需要对中控数组 map 进行扩容,再将原 map 中的数组指针拷贝过来即可,效率比较高

deque 中的随机访问:

1.(下标 -前预留位)/单个数组长度 获取对应小数组位置

2.(下标 -前预留位)%单个数组长度 获取其在小数组中的对应下标

示例
假设我们有一个std::deque,它由3个数组组成,每个数组可以存储4个元素。现在我们想要访问下标为8的元素:
1.确定数组索引:
设前预留位为0(即没有预留空间),
则数组索引为(8-0)/4=2。这意味着我们要找的元素在第三个数组中。
2.确定数组内偏移:
数组内偏移为(8-0)%4=0。这意味着元素是第三个数组的第一个元素

由此可见,单个数组大小(缓冲区大小)需要定长,否则访问时计算会比较麻烦,但长度定长后,会引发中间位置插入删除效率低的问题

对此 sGI 版的 STL 选择牺牲中间位置插入,提高下标随机访问速度,令小数组定长,这也是将它作为栈和队列默认底层容器的原因之,因为栈和队列不需要对中间进行操作

为了维护其“整体连续”以及随机访问的假象,落在了deque的迭代器身上

5.2迭代器设计

因此deque的迭代器设计就比较复杂,如下图所示:  

cur 指向当前需要的数据位置
first 指向 buffer 数组起始位置
last 指向 buffer 数组终止位置
node 反向指向中控数组

这个迭代器还是一个随机迭代器,因此可以使用 std::sort

无论是 deque 还是 list,直接排序的效率都不如借助vector间接排序效率高,主要原因还是因为影响快排效率的因素主要是对各个位置数据的访问效率
 

5.3deque 的缺点

中间位置插入删除比较麻烦,可以令小数组长度不同解决问题,不过此时影响随机访问效率
结构设计复杂,且不如vector和list 极致

致命缺陷:不适合遍历,迭代器需要频繁检测是否移动某段小空间的边界,效率很低
凑巧的是,栈和队列可以完美避开所有缺陷,全面汲取其优点,因此双端队列为容器适配器的默认底层容器

 

对于这种中庸且复杂的容器,只需要做简单了解就行了

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

sqli-labs靶场搭建

下载了一个phpstudy进行搭靶场搭建 然后打开phpstudy安装好php,mysql等环境 正式sqli-labs靶场搭建 第一步&#xff1a;下载源码&#xff1a;https://codeload.github.com/Audi-1/sqli-labs/zip/master 解压后放进网站根目录&#xff0c;进到 sqli-labs的文件夹下&#xff0…

windows C++ 并行编程-异步代理库概述

异步代理库&#xff08;简称代理库&#xff09;提供了一个编程模型&#xff0c;该模型可提高支持并发的应用程序开发的可靠性。 代理库是一个 C 模板库&#xff0c;为粗粒度数据流和管道任务提升了基于角色的编程模型和进程内消息传递。 代理库构建在并发运行时的计划和资源管理…

Windows系统通过部署wsl + Goland进行跨平台开发

1.背景 近期项目中因为用到了 Golang库中的 "log/syslog" 包,而这个包是禁止在windows平台上编译的. 并且在windows环境上开发也会有诸多不便,如执行makefile文件的make命令,本地开发环境中docker,etcd,redis的搭建等等,而这些通过部署wsl去搭建一个linux环境就很可以…

如何使用下拉字段创建WordPress表单(简单方法)

许多网站所有者在收集用户输入时&#xff0c;都会因为表单过长而让用户感到压迫。 下拉列表字段通过提供一个简洁的选项列表&#xff0c;使表单变得更简单。这意味着它们可以提高表单完成率&#xff0c;并改善用户体验。 在本文中&#xff0c;我们将向您展示如何创建带有下拉…

Kubernetes从零到精通(11-CNI网络插件)

Kubernetes网络模型 Kubernetes的网络模型&#xff08;Kubernetes Networking Model&#xff09;旨在提供跨所有节点、Pod和服务的统一网络连接。它的核心理念是通过统一的网络通信规则&#xff0c;保证集群中的所有组件能够顺畅地相互通信。Kubernetes网络模型主要有以下几个关…

专业学习|随机规划概观(性质、针对问题与分类)

一、随机规划概观 随机规划&#xff08;Stochastic Programming&#xff09;是一种用于处理决策问题中的不确定性的优化方法。它能够在决策过程中考虑到未来的不确定性&#xff0c;从而帮助找到在不同情境下都能较好表现的解决方案。以下是随机规划能解决的一些主要问题以及它的…

阿里巴巴搜索API返回值:电商市场竞争的新武器含

阿里巴巴搜索API返回值在电商市场竞争中扮演着至关重要的角色&#xff0c;它为企业提供了深入了解市场、分析竞争对手的宝贵资源。以下是对阿里巴巴搜索API返回值及其在电商市场竞争中应用的详细解析&#xff0c;并附上示例代码。 一、阿里巴巴搜索API返回值概述 阿里巴巴搜索…

超大酒店司机收布草-酒店分层管理--酒店布草洗涤

一、大酒店布草分层管理 1. 提高效率 - 对布草进行分层&#xff0c;可以更有针对性地安排收集和分发流程&#xff0c;减少混乱和等待时间&#xff0c;提高整体运营效率。 2. 质量控制 - 不同层级的布草可能有不同的质量标准和使用场景。分层管理有助于确保每个层级的…

2024年第五届“华数杯”全国大学生数学建模竞赛 A题详细思路+详细matlab代码

没有更新完之前,专栏价格为59,更新完毕之后恢复到99. 专栏内包含2024年所有数学建模比赛思路和代码,有些重要比赛着重更新(华数杯、国赛、美赛),小比赛可能会有chatgpt4更新,只需订阅一次。有些文章没有完整代码,请到专栏内查找最新代码和思路。如果比赛结束后没有更新…

Web后端开发技术:RESTful 架构详解

RESTful 是一种基于 REST&#xff08;表述性状态转移&#xff0c;Representational State Transfer&#xff09;架构风格的 API 设计方式&#xff0c;通常用于构建分布式系统&#xff0c;特别是在 Web 应用开发中广泛应用。REST 是一种轻量级的架构模式&#xff0c;利用标准的 …

大语言模型超参数调优:开启 AI 潜能的钥匙

前言 在人工智能的广袤领域中&#xff0c;大语言模型&#xff08;LLM&#xff09;凭借其强大的实力&#xff0c;不断重塑着我们对机器理解语言的认知。然而&#xff0c;要使这些模型在特定应用场景中发挥最大效能&#xff0c;关键在于巧妙调整其超参数。本文将引领你深入探究 …

【SSM-Day2】第一个SpringBoot项目

运行本篇中的代码&#xff1a;idea专业版或者idea社区版本&#xff08;2021.1~2022.1.4&#xff09;->这个版本主要是匹配插件spring boot Helper的免费版(衰) 【SSM-Day2】第一个SpringBoot项目 框架->Spring家族框架快速上手Spring BootSpring Boot的作用通过idea创建S…

Kettle报错:使用mysql向hive中插入数据只能插入两条的错误

错误展示 我们在用kettle&#xff0c;使用mysql向hive中插入数据的时候&#xff0c;创建好了一个转换&#xff0c;里面的操作也全部完成了之后&#xff0c;在执行时爆出一下错误 例如我这里写入的表输入为&#xff1a; 表输出为&#xff1a; 解决办法 看起来是一点问题也没有…

HFSS 常见仿真警告、报错及bug处理

目录 引言提示信息警告信息报错信息导入csv文件报错 内部bugHFSS切换工程文件&#xff0c;视图窗口卡顿 引言 本文主要用于收录HFSS仿真中常见的错误及处理方法。欢迎大家在评论区贴出自己的报错信息&#xff0c;一起讨论分享。 提示信息 提示信息&#xff1a;Port 7 suppor…

C++调用C# DLL之踩坑记录

C是非托管代码&#xff0c;C#则是托管代码&#xff0c;无法直接调用 CLR的介绍见CLR简介 MSDN提到了两种非托管-托管的交互技术&#xff1a;CLR Interop和COM Interop 后者要将C# 类库注册为COM组件&#xff0c;本文只探讨CLR&#xff0c;要通过C CLR写中间层代码 方式一&…

htaccess转换nginx工具

115工具网为您提供htaccess与nginx在线转换,apache伪静态文件转为nginx重写规则,htaccess伪静态规则换nginx,apache RewriteRule转rewrite,apache伪静态文件转nginx重写,apache转nginx重写规则&#xff0c;本工具支持所有的htaccess伪静态、基本的配置规则、重定向等转换为ngin…

Golang开发的OCR-身份证号码识别(不依赖第三方)

身份证号码识别&#xff08;golang&#xff09; 使用golang的image库写的身份证号码识别&#xff0c;还有用了一个resize外部库&#xff0c;用来更改图片尺寸大小&#xff0c;将每个数字所在的图片的大小进行统一可以更好的进行数字识别&#xff0c;库名 &#xff1a;“github…

C语言 ——— 编写函数,判断一个整数是否是回文整数

目录 题目要求 代码实现 题目要求 编写一个函数&#xff0c;用来判断一个整数是否是回文整数&#xff0c;如果是回文整数就返回 true &#xff0c;如果不是就返回 false 举例说明&#xff1a; 输入&#xff1a;121 输出&#xff1a;true 输入&#xff1a;1321 输出&#xf…

怎么把文件生成二维码活码?支持生成多种格式文件的二维码教程

怎么把文件做成二维码分享给其他人预览或下载呢&#xff1f;现在使用二维码来展示或者分享文件的使用场景越来越多&#xff0c;这种方式可以帮助其他人更快的获取文件内容&#xff0c;有利于提升文件传输的效率。二维码可以长期存储文件&#xff0c;获取文件会更加的灵活方便&a…

翻转对00

题目链接 翻转对 题目描述 注意点 给定数组的长度不会超过50000输入数组中的所有数字都在32位整数的表示范围内 解答思路 本题与区间和的个数类似&#xff0c;都是使用归并排序统计满足题意的数量&#xff0c;归并排序后可以有效减少比较的数量归并排序的思路为&#xff1…