【C++从0到王者】第十六站:stack和queue的使用

文章目录

  • 一、stack的使用
    • 1.stack的介绍
    • 2.stack的使用
  • 二、queue的使用
    • 1.queue的护额晒
    • 2.queue的使用
  • 三、stack和queue相关算法题
    • 1.最小栈
    • 2.栈的压入、弹出序列
    • 3.逆波兰表达式
    • 4.两个栈实现一个队列
    • 5.用两个队列实现栈
    • 6.二叉树的层序遍历
      • 1.双队列
      • 2.用一个变量levelSize去控制
    • 7.二叉树的层序遍历Ⅱ

一、stack的使用

1.stack的介绍

在这里插入图片描述

栈是一种容器适配器,专门设计用于在 LIFO 上下文中操作(后进先出),其中元素只从容器的一端插入和提取。

栈被实现为容器适配器,这些类使用特定容器类的封装对象作为其底层容器,提供一组特定的成员函数来访问其元素。 元素从特定容器的“后端”被推入/弹出,这被称为栈的顶部。

底层容器可以是任何标准的容器类模板,也可以是其他一些专门设计的容器类。容器应该支持以下操作:empty、size、back、push_back、pop_front

标准容器类 vector、deque 和 list 满足这些要求。默认情况下,如果没有为特定的堆栈类实例化指定容器类,则使用标准容器 deque。

这里我们需要知道的是,Container适配器就是一个现有的容器进行的一个转换
也就是说,适配器的本质就是一种复用

2.stack的使用

我们先看stack的接口有哪些
在这里插入图片描述

如上所示,这里其实我们一看就已经猜出了七七八八了。因为与前面是string、vector、list是十分相似的。只要结合它的先进先出的特性,我们就知道每个函数都是什么意思了。

对于stack的使用是非常简单的

void test_stack()
{stack<int> st1;st1.push(1);st1.push(2);st1.push(3);st1.push(4);while (!st1.empty()){cout << st1.top() << " ";st1.pop();}cout << endl;
}

在这里插入图片描述

二、queue的使用

1.queue的护额晒

在这里插入图片描述

队列是一种容器适配器,专门设计用于在FIFO上下文中操作(先进先出),其中将元素插入容器的一端并从另一端提取。

队列是作为容器适配器实现的,容器适配器是使用特定容器类的封装对象作为其底层容器的类,提供一组特定的成员函数来访问其元素。元素被推入特定容器的“后面”,并从其“前面”弹出。

底层容器可以是标准容器类模板之一,也可以是其他专门设计的容器类。该底层容器应至少支持以下操作:empty、size、front、back、push_back、pop_back。

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

2.queue的使用

同样的,我们先来看queue的接口有哪些
在这里插入图片描述

在这些接口中,和stack是类似的,只要我们知道了queue的特性是先进先出。我们就能很轻松的推测出每个接口的意思

对于queue的使用是很简单的

void test_queue()
{queue<int> q;q.push(1);q.push(2);q.push(3);q.push(4);while (!q.empty()){cout << q.front() << " ";q.pop();}cout << endl;
}

在这里插入图片描述

三、stack和queue相关算法题

1.最小栈

题目链接:最小栈

题目解析:在这道题中,我们为了模拟这种情况,我们可以在成员变量中定义两个栈,一个栈作为正常的出入数据使用,一个栈用来存储最小值。
首先是构造函数,由于我们的成员变量都是自定义类型,所以会自动调用他们的默认构造函数,即他们也会走初始化列表,所以默认构造函数我们是可以直接不用管的。甚至于我们可以直接删掉题目给的构造函数,因为我们不写,编译器自己生成一个。
其次我们的大逻辑是这样的,当我们最小栈为空的时候,我们的最小栈是需要入一个数据的,当我们将要插入的元素是小于等于最小栈要插入的元素的时候,我们会将这个元素给入最小栈
当我们pop的时候,我们也是同理的,如果我们要删除的数据等于最小栈的栈顶元素,那么就也要删除最小栈的栈顶元素。

class MinStack {
public:MinStack(){}void push(int val) {st.push(val);if(min.empty()||(val<=min.top())){min.push(val);}}void pop() {int val=st.top();st.pop();if(val==min.top()){min.pop();}}int top() {return st.top();}int getMin() {return min.top();}
private:stack<int> st;stack<int> min;
};/*** Your MinStack object will be instantiated and called as such:* MinStack* obj = new MinStack();* obj->push(val);* obj->pop();* int param_3 = obj->top();* int param_4 = obj->getMin();*/

2.栈的压入、弹出序列

题目链接:栈的压入弹出序列

题目解析:我们可以使用一个栈来模拟它的入栈出栈逻辑,只要顺着它的思路最终我们的这个栈是空栈的话,那么就说明是匹配的,否则不匹配

class Solution {
public:/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可** * @param pushV int整型vector * @param popV int整型vector * @return bool布尔型*/bool IsPopOrder(vector<int>& pushV, vector<int>& popV) {// write code herestack<int> st;int n=pushV.size();int j=0;for(int i=0;i<n;i++){st.push(pushV[i]);while(!st.empty() && st.top()==popV[j]){st.pop();j++;}            }return st.empty();}
};

3.逆波兰表达式

题目链接:逆波兰表达式

再谈这道题之前,我们应该先知道什么是逆波兰表达式,我们正常的都是中缀表达式,即3+2*5这种的,都被称之为中缀表达式。

而中缀表达式在计算机中是很难进行运算的。我们需要先将其转换为后缀表达式,前文所说的中缀表达式转化为后缀表达式后应该为3 2 5 * +。后缀表达式的特点就是操作数的顺序不变,而操作符的顺序按照优先级进行了重排。

我们先来看一下后缀运算符是如何进行运算的:

  1. 操作数入栈
  2. 如果是操作符,取出栈顶的两个元素进行计算,计算结果放入栈中

在这里插入图片描述

那么如何使得中缀转为后缀呢?

  1. 操作数输出(即将操作数放到一个容器中)
  2. 操作符入栈 : ①栈为空,当前操作符比栈顶的优先级高,继续入栈 ②栈不为空,且当前操作符比栈顶的优先级低或者相等,则输出栈顶操作符(因为运算符的优先级只与它相邻的操作符有关,是相对的,如果后面出现了一个更高的操作符,我们无法确定后面是否还有更高的操作符,反而是如果有一个相对较低的操作符,那么前两个肯定是可以进行运算的)
  3. 表达式结束后,依次出栈顶的操作符
  4. 注意,有可能会在转换的中间出现连续出操作符的情况,即栈里面已经存储了好几个运算符了,下面的一个运算符要比好几个都要低,就要连续出好几个运算符

比如说2+4-1*3,这个中缀表达式,按照上面的规则可以化为
在这里插入图片描述

上面都是正常情况的下的处理,但是还有时候会出现括号的影响。

这里可以考虑加上一个特殊标记,当我们这个标记生效时,就代表进入括号内了。或者在这里走一个递归也是可以的。递归的方法就是在遇到括号的时候,我们将括号里面的运算符就需要放入一个新的栈中了。相当于我们只需要让括号返回一个结果就可以了。但是数据还是输出在原来的顺序表中的

我们在回过头来看这道题,我们有了上面的分析,就很容易写出下面代码了。

class Solution {
public:int evalRPN(vector<string>& tokens) {stack<int> st;for(auto& str : tokens){if(str=="+"||str=="-"||str=="*"||str=="/"){int right=st.top();st.pop();int left=st.top();st.pop();switch(str[0]){case '+':st.push(left+right);break;case '-':st.push(left-right);break;case '*':st.push(left*right);break;case '/':st.push(left/right);break;}}else{st.push(stoi(str));}}return st.top();}
};

4.两个栈实现一个队列

题目链接:两个栈实现一个队列

对于这个题,我们在之前也已经做过一次分析了,只不过上一次是用C语言手撕了一个栈来实现了。而现在呢,我们有C++的库了,因此我们就可以直接使用C++的库来完成这件事。

class MyQueue {
public:MyQueue() {}void push(int x) {_push.push(x);}int pop() {if(_pop.empty()){while(!_push.empty()){int val=_push.top();_push.pop();_pop.push(val);}}int val=_pop.top();_pop.pop();return val;}int peek() {if(_pop.empty()){while(!_push.empty()){int val=_push.top();_push.pop();_pop.push(val);}}return _pop.top();}bool empty() {return (_push.empty()&&_pop.empty());}
private:stack<int> _push;stack<int> _pop;
};/*** Your MyQueue object will be instantiated and called as such:* MyQueue* obj = new MyQueue();* obj->push(x);* int param_2 = obj->pop();* int param_3 = obj->peek();* bool param_4 = obj->empty();*/

5.用两个队列实现栈

题目链接:用两个队列实现栈

这道题与上面的题类似,我们也是曾经使用C语言做过,不过由于C语言没有轮子,就需要我们自己造轮子,有了很多的麻烦。

class MyStack {
public:MyStack() {}void push(int x) {if(q1.empty()){q2.push(x);}else{q1.push(x);}}int pop() {if(q1.empty()){while(q2.size()>1){int val=q2.front();q2.pop();q1.push(val);}int val=q2.front();q2.pop();return val;}else{while(q1.size()>1){int val=q1.front();q1.pop();q2.push(val);}int val=q1.front();q1.pop();return val;}}int top() {if(q1.empty()){return q2.back();}else{return q1.back();}}bool empty() {return (q1.empty()&&q2.empty());}
private:queue<int> q1;queue<int> q2;
};/*** 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();*/

6.二叉树的层序遍历

题目链接:二叉树的层序遍历

对于这道题,如果不是用C++来完成的话,用C去完成的话是非常难的。不仅我们要确保我们造的轮子是正确的,而且还有很多细节需要进行处理,但我们如果使用C++的话可以极大的简化很多操作

1.双队列

我们可以使用两个队列去完成这件事,一个队列用来存储结点指针,一个队列用来存储该节点处于哪个层。这样我们就可以知道哪个结点是那个层的了,自然我们就很容易的得知层序遍历了。

2.用一个变量levelSize去控制

这种思路是比较奇妙的,我们使用一个变量来确定当前该层有多少个结点,用一个队列来存储结点,然后就是层序遍历的基本套路,每出一个结点,带来两个孩子。将每一层的数据存储在一个vector中,然后一层结束后将这一层的vector插入到vector<vector<int>>中。

/*** 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) {queue<TreeNode*> q;int levelSize=0;if(root){q.push(root);levelSize=1;}vector<vector<int>> vv;while(!q.empty()){vector<int> v;for(int i=0;i<levelSize;i++){TreeNode* front=q.front();q.pop();v.push_back(front->val);if(front->left){q.push(front->left);}if(front->right){q.push(front->right);}}vv.push_back(v);levelSize=q.size();}return vv;}
};

7.二叉树的层序遍历Ⅱ

题目链接:二叉树的层序遍历Ⅱ

对于这道题目,我们可以注意到他是让倒着遍历的。和前一道题基本是一样的,只是改变了vv的顺序。即相当于将vv给逆序。那么这道题就太简单了,直接将前面这道题给拷贝过来,然后调用库里面的reverse即可

/*** 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>> levelOrderBottom(TreeNode* root) {queue<TreeNode*> q;int levelSize=0;if(root){q.push(root);levelSize=1;}vector<vector<int>> vv;while(!q.empty()){vector<int> v;for(int i=0;i<levelSize;i++){TreeNode* front=q.front();q.pop();v.push_back(front->val);if(front->left){q.push(front->left);}if(front->right){q.push(front->right);}}vv.push_back(v);levelSize=q.size();}reverse(vv.begin(),vv.end());return vv;}
};

好了,本期内容就到这里了
如果对你有帮助的话,不要忘记点赞加收藏哦!!!

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

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

相关文章

ECharts 折线图使用相关

一、折线图堆叠设置为不堆叠的方法 官网是这样的&#xff0c;但是不需要这种堆叠形式的如下图&#xff1a; 即&#xff1a;第2条数据值 第1条数据值 第2条数据值 ​​​​​​​ 第3条数据值 第2条数据值 第3条数据值 需要改成实际值展示&#xff0c;如下图&#xff1a; 只…

数据结构之栈和队列---c++

栈和队列的简单介绍 栈 栈是一个“先进后出”结构 队列 入队演示 队列是一种“先进先出”的结构 出队演示 接下来我们开始本次的内容 栈实现队列 分析 1.我们可以老老实实的写一个栈然后将所有的接口函数实现出来&#xff0c;最后再进行实现队列&#xff0c;但是显然…

【雕爷学编程】Arduino动手做(193)---移远 BC20 NB+GNSS模块7

37款传感器与模块的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&#x…

【数学建模学习(9):模拟退火算法】

模拟退火算法(Simulated Annealing, SA)的思想借 鉴于固体的退火原理&#xff0c;当固体的温度很高的时候&#xff0c;内能比 较大&#xff0c;固体的内部粒子处于快速无序运动&#xff0c;当温度慢慢降 低的过程中&#xff0c;固体的内能减小&#xff0c;粒子的慢慢趋于有序&a…

空地协同智能消防系统——无人机、小车协同

1 题目 1.1 任务 设计一个由四旋翼无人机及消防车构成的空地协同智能消防系统。无人机上安装垂直向下的激光笔&#xff0c;用于指示巡逻航迹。巡防区域为40dm48dm。无人机巡逻时可覆盖地面8dm宽度区域。以缩短完成全覆盖巡逻时间为原则&#xff0c;无人机按照规划航线巡逻。发…

2019年09月 Python(一级)真题解析#中国电子学会#全国青少年软件编程等级考试

一、单选题 第1题 关于Python的编程环境,下列的哪个表述是正确的? A:Python的编程环境是图形化的; B:Python只有一种编程环境ipython; C:Python自带的编程环境是IDLE; D:用windows自带的文本编辑器也可以给Python编程?,并且也可以在该编辑器下运行; 正确答案…

自动驾驶传感器选型

360的场景&#xff0c;避免有盲区&#xff0c;长距离 Lidar&#xff08;激光雷达&#xff09; 典型特点一圈一圈的&#xff0c;轮廓和很高的位置精度 禾赛的机械雷达 速腾的固态雷达 固态雷达是车规级的&#xff0c;车规级的意思是可以装到量产车上 Radar&#xff08;毫米…

门面模式(C++)

定义 为子系统中的一组接口提供一个一致(稳定) 的界面&#xff0c;Facade模式定义了一个高层接口&#xff0c;这个接口使得这一子系统更加容易使用(复用)。 应用场景 上述A方案的问题在于组件的客户和组件中各种复杂的子系统有了过多的耦合&#xff0c;随着外部客户程序和各子…

数据结构—哈夫曼树及其应用

5.6哈夫曼树及其应用 5.6.1哈夫曼树的基本概念 路径&#xff1a;从树中一个结点到另一个结点之间的分支构成这两个结点间的路径。 结点的路径长度&#xff1a;两结点间路径上的分支数。 树的路径长度&#xff1a;从树根到每一个结点的路径长度之和。记作 TL 结点数目相同的…

【jvm】jvm整体结构(hotspot)

目录 一、说明二、java代码的执行流程三、jvm的架构模型3.1 基于栈式架构的特点3.2 基于寄存器架构的特点 一、说明 1.hotspot vm是目前市场上高性能虚拟机的代表作之一 2.hotspot采用解释器与即时编译器并存的架构 3.java虚拟机是用来解释运行字节码文件的&#xff0c;入口是字…

混合云环境实现K8S可观测的6大策略

2023年&#xff0c;原生云应用及平台发展迅猛。大量企业都在努力发挥其应用程序的最大潜力&#xff0c;以确保极致的用户体验并推动业务增长。 混合云环境的兴起和容器化技术(如Kubernetes)的采用彻底改变了现代应用程序的开发、部署和扩展方式。 在这个数字舞台上&#xff0c;…

命令模式 Command Pattern 《游戏设计模式》学习笔记

对于一般的按键输入&#xff0c;我们通常这么做&#xff0c;直接if按了什么键&#xff0c;就执行相应的操作 在这里我们是将用户的输入和程序行为硬编码在一起&#xff0c;这是我们很自然就想到的最快的做法。 但是如果这是一个大型游戏&#xff0c;往往我们需要实现一个按键…

研发工程师玩转Kubernetes——hostPath

有别于《研发工程师玩转Kubernetes——emptyDir》一文中介绍的emptyDir&#xff0c;hostPath可以在同一个Node的不同Pod间共享卷。 下面的清单文件利用了Pod亲和性&#xff0c;让Pod集中到一个Node上。 apiVersion: apps/v1 kind: Deployment metadata:name: hostpath-deploy…

中间人攻击

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、启动 Kali Linux准备 Apache 服务器使用Ettercap插件进行欺骗使用 Wireshark 捕获 Https 流量总结 前言 您可能已经熟悉中间人攻击&#xff1a;攻击者通过…

网络安全(秋招)如何拿到offer?(含面试题)

以下为网络安全各个方向涉及的面试题&#xff0c;星数越多代表问题出现的几率越大&#xff0c;祝各位都能找到满意的工作。 注&#xff1a;本套面试题&#xff0c;已整理成pdf文档&#xff0c;但内容还在持续更新中&#xff0c;因为无论如何都不可能覆盖所有的面试问题&#xf…

C语言预处理命令 #error 学习

#error命令是C/C语言的预处理命令之一&#xff0c;当预处理器预处理到#error命令时将停止编译并输出用户自定义的错误消息。 如下代码输出数字1000&#xff0c;如果加了 #error&#xff0c;构建时不会通过&#xff0c;提示出错如下&#xff1b; 这可能在大型项目中比较有用&am…

8.4一日总结

1.远程仓库的提交方式(免密提交) a.ssh:隧道加密传输协议,一般用来登录远程服务器 b.使用 git clone 仓库名 配置(生成公私钥对) ssh-Keygen [-t rsa -C 邮箱地址] 通过执行上述命令,全程回车,就会在~/.ssh/id_rsa(私钥)和id_rsa.pub(公钥),私钥是必须要保存好的,并不能…

TeeChart NET for MAUI Crack

TeeChart NET for MAUI Crack 跨平台图表-移动或桌面应用程序的核心图表代码相同。 图表集合-60多种图表类型和50多种财务和统计指标。 图表类型 60多种2D和3D图表类型以及多种组合&#xff0c;包括&#xff1a; 标准&#xff1a;线条(条形)、条形、区域、饼图、快线、点(散点…

【Java可执行命令】(十七)JVM运行时信息动态维护工具 jinfo:一个维护 JVM 相关的配置参数和系统属性的工具,辅助故障排除、诊断和优化 ~

Java可执行命令之jinfo 1️⃣ 概念2️⃣ 优势和缺点3️⃣ 使用3.1 语法格式3.2 -flags&#xff1a;查看进程的启动参数3.3 -sysprops&#xff1a;查看进程的系统属性3.4 -flag < name>&#xff1a;查看特定虚拟机参数的值3.5 -flag [/-]< name>&#xff1a;启用或禁…

2023年8月份华为H12-811更新了

801、[单选题]178/832、在系统视图下键入什么命令可以切换到用户视图? A quit B souter C system-view D user-view 试题答案&#xff1a;A 试题解析&#xff1a;在系统视图下键入quit命令退出到用户视图。因此答案选A。 802、[单选题]“网络管理员在三层交换机上创建了V…