当前位置: 首页 > news >正文

C++习题精选(4)—— 栈

目录

  • 1. 最小栈
  • 2. 栈的压入弹出序列
  • 3. 逆波兰表达式求值

1. 最小栈

题目描述:设计一个支持 push ,pop ,top 操作,并能在常数时间内检索到最小元素的栈。实现 MinStack 类:
MinStack() 初始化堆栈对象。
void push(int val) 将元素val推入堆栈。
void pop() 删除堆栈顶部的元素。
int top() 获取堆栈顶部的元素。
int getMin() 获取堆栈中的最小元素。

示例 1:
输入:
[“MinStack”,“push”,“push”,“push”,“getMin”,“pop”,“top”,“getMin”]
[[],[-2],[0],[-3],[],[],[],[]]

输出:
[null,null,null,null,-3,null,0,-2]

解释:
MinStack minStack = new MinStack();
minStack.push(-2);
minStack.push(0);
minStack.push(-3);
minStack.getMin(); --> 返回 -3.
minStack.pop();
minStack.top(); --> 返回 0.
minStack.getMin(); --> 返回 -2.

提示:
-231 <= val <= 231 - 1
pop、top 和 getMin 操作总是在 非空栈 上调用
push, pop, top, and getMin最多被调用 3 * 104 次

题目链接: https://leetcode.cn/problems/min-stack/?clienttype=8&version=7.41.0.106&from=win32_yunguanjia&channel=00000000000000000000000040000001&privilege=&pri_extra=

解题思路: 再创建一个栈出来,则有两个栈——栈A和栈B。原数据每次都往栈A中插入,而栈B为空时插入,或者当前插入数据等于或小于栈B栈顶的数据小时才插入。当弹出数据时,栈A每次都弹出栈顶,栈B栈顶的数据与栈A弹出的数据进行比较,若相等则弹出。所以,栈B栈顶的数据就是当前栈A中最小的数据。

代码如下:

class MinStack {
private:stack<int> A;stack<int> B;
public:MinStack() {}void push(int val) {// 栈A每次都压入数据A.push(val);// 当栈B为空时,或者栈A的栈顶数据小于等于栈B栈顶的数据,就往栈B中压入数据if (B.empty() || B.top() >= A.top())B.push(val);}void pop() {// 当栈B栈顶的数据等于栈A栈顶的数据时,栈B弹出数据if (A.top() == B.top())B.pop();// 栈A每次都弹出数据A.pop();}int top() {// 栈A栈顶的数据就是当前栈顶的数据return A.top();}int getMin() {// 栈B栈顶的数据就是当前栈中最小的数据return B.top();}
};

2. 栈的压入弹出序列

题目描述:输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的弹出顺序。假设压入栈的所有数字均不相等。例如序列1,2,3,4,5是某栈的压入顺序,序列4,5,3,2,1是该压栈序列对应的一个弹出序列,但4,3,5,1,2就不可能是该压栈序列的弹出序列。
输入限制:
1. 0<=pushV.length == popV.length <=1000
2. -1000<=pushV[i]<=1000
3. pushV 的所有数字均不相同

示例1
输入:
[1,2,3,4,5],[4,5,3,2,1]
返回值:true
说明:可以通过push(1)=>push(2)=>push(3)=>push(4)=>pop()=>push(5)=>pop()=>pop()=>pop()=>pop()这样的顺序得到[4,5,3,2,1]这个序列,返回true

示例2
输入:[1,2,3,4,5],[4,3,5,1,2]
返回值:false
说明:由于是[1,2,3,4,5]的压入顺序,[4,3,5,1,2]的弹出顺序,要求4,3,5必须在1,2前压入,且1,2不能弹出,但是这样压入的顺序,1又不能在2之前弹出,所以无法形成的,返回false

题目链接:https://www.nowcoder.com/share/jump/8812161311718109046298

**解题思路:**新建一个 tmp 栈通过循环来模拟栈的压入弹出的过程。当tmp栈为空或者tmp栈栈顶的数据不等于弹出数组的数据时,则通过压入数组压入数据。定义两个下标变量,跟踪压入和弹出数组当前元素的位置。当压入数组下标超出之后便结束循环。此时,若tmp数组中没有元素,则证明该弹出序列正确,否则该弹出序列错误。

代码如下:

class Solution {
public:bool IsPopOrder(vector<int>& pushV, vector<int>& popV) {// 存储模拟过程stack<int> tmp;// 所需变量int push_size = 0, pop_size = 0;// 循环计算while (push_size < pushV.size()){tmp.push(pushV[push_size]);++push_size;// 如果tmp栈不为空并且栈顶数据与弹出数组当前元素相同,tmp栈弹出数据while (!tmp.empty() && tmp.top() == popV[pop_size]){tmp.pop();++pop_size;}}// 结果判断return tmp.empty();}
};

3. 逆波兰表达式求值

题目描述:给你一个字符串数组 tokens ,表示一个根据 逆波兰表示法 表示的算术表达式。请你计算该表达式。返回一个表示表达式值的整数。

注意:
有效的算符为 ‘+’、‘-’、‘*’ 和 ‘/’ 。
每个操作数(运算对象)都可以是一个整数或者另一个表达式。
两个整数之间的除法总是 向零截断 。
表达式中不含除零运算。
输入是一个根据逆波兰表示法表示的算术表达式。
答案及所有中间计算结果可以用 32 位 整数表示。

示例 1:
输入:tokens = [“2”,“1”,“+”,“3”,“*”]
输出:9
解释:该算式转化为常见的中缀算术表达式为:((2 + 1) * 3) = 9

示例 2:
输入:tokens = [“4”,“13”,“5”,“/”,“+”]
输出:6
解释:该算式转化为常见的中缀算术表达式为:(4 + (13 / 5)) = 6

示例 3:
输入:tokens = [“10”,“6”,“9”,“3”,“+”,“-11”,““,”/“,””,“17”,“+”,“5”,“+”]
输出:22
解释:该算式转化为常见的中缀算术表达式为:
((10 * (6 / ((9 + 3) * -11))) + 17) + 5
= ((10 * (6 / (12 * -11))) + 17) + 5
= ((10 * (6 / -132)) + 17) + 5
= ((10 * 0) + 17) + 5
= (0 + 17) + 5
= 17 + 5
= 22

提示:
1 <= tokens.length <= 104
tokens[i] 是一个算符(“+”、“-”、“*” 或 “/”),或是在范围 [-200, 200] 内的一个整数

逆波兰表达式:
逆波兰表达式是一种后缀表达式,所谓后缀就是指算符写在后面。
平常使用的算式则是一种中缀表达式,如 ( 1 + 2 ) * ( 3 + 4 ) 。
该算式的逆波兰表达式写法为 ( ( 1 2 + ) ( 3 4 + ) * ) 。
逆波兰表达式主要有以下两个优点:
去掉括号后表达式无歧义,上式即便写成 1 2 + 3 4 + * 也可以依据次序计算出正确结果。
适合用栈操作运算:遇到数字则入栈;遇到算符则取出栈顶两个数字进行计算,并将结果压入栈中

题目链接:link

**解题思路:**大家日常生活中使用的都是中缀表达式,也就是运算符在中间,操作数在两边。然后运算时根据操作符的优先级和结合顺序进行计算。而后缀表达式,则是已经排序好了,根据顺序计算就行,遇到运算符就把前第一个数当右操作数,前第二个数当作做操作数进行运算。则我们使用一个栈,当遇到操作数时进行压栈操作,当遇到操作符时取出栈顶的前两个数据进行运算,然后把结果压入栈中。如此往复,则最终栈顶的数据就是该表达式的结果。由于传入的数据是以字符串的形式传入的,则需要把字符串转换为整形,可以自己设计函数也可以使用string类库中的函数——void stoi(const string& s)。作者判断当前字符串是否为运算符是根据ASCII码表的值进行比较的。

代码如下:

class Solution {
public:int evalRPN(vector<string>& tokens) {// 使用一个int栈stack<int> i_stack;// 使用迭代器遍历数组for (auto& val : tokens){// 是运算符就运算if (val[0] < 48 && val[1] < 48){// 向栈中取两个操作数int n2 = i_stack.top();i_stack.pop();int n1 = i_stack.top();i_stack.pop();// 根据符号进行运算switch (val[0]){case '+':i_stack.push(n1 + n2);break;case '-':i_stack.push(n1 - n2);break;case '*':i_stack.push(n1 * n2);break;case '/':i_stack.push(n1 / n2);break;}}else  // 不是运算符就压栈{i_stack.push(stoi(val));}}// 完成处理后,栈顶就是运算结果return i_stack.top();}
};

相关文章:

  • Mybatis05-一对多和多对一处理
  • 最大二叉树-力扣
  • 从零实现ChatGPT:第四章在无标签数据上预训练
  • 手写节流防抖函数
  • 多个线程多个锁:如何确保线程安全和避免竞争条件
  • Python pandas openpyxl excel合并单元格,设置边框,背景色
  • 在 Linux 系统上安装 Android NDK
  • 呼叫中心系统的国产化替代方案
  • 【Flutter】 TextField限制长度时, 第三方手写输入法、ios原始拼音输入法输入被吞问题
  • swift微调牧歌数据电商多模态大语言模型
  • 【Spring6】1-12章源码级深入详解 IoC
  • 网络安全(补充)
  • 外卖抢单神器
  • 重学java 66.IO流 转换流
  • Linux-笔记 设备树插件
  • ES6指北【2】—— 箭头函数
  • 2017-09-12 前端日报
  • Asm.js的简单介绍
  • Bootstrap JS插件Alert源码分析
  • DOM的那些事
  • electron原来这么简单----打包你的react、VUE桌面应用程序
  • JSONP原理
  • mysql 数据库四种事务隔离级别
  • Spark学习笔记之相关记录
  • springboot_database项目介绍
  • sublime配置文件
  • ubuntu 下nginx安装 并支持https协议
  • windows下如何用phpstorm同步测试服务器
  • 从零到一:用Phaser.js写意地开发小游戏(Chapter 3 - 加载游戏资源)
  • 分享几个不错的工具
  • 前端技术周刊 2018-12-10:前端自动化测试
  • 人脸识别最新开发经验demo
  • 腾讯优测优分享 | 你是否体验过Android手机插入耳机后仍外放的尴尬?
  • 一加3T解锁OEM、刷入TWRP、第三方ROM以及ROOT
  • 一些css基础学习笔记
  • shell使用lftp连接ftp和sftp,并可以指定私钥
  • # C++之functional库用法整理
  • $ is not function   和JQUERY 命名 冲突的解说 Jquer问题 (
  • (33)STM32——485实验笔记
  • (8)STL算法之替换
  • (delphi11最新学习资料) Object Pascal 学习笔记---第14章泛型第2节(泛型类的类构造函数)
  • (Matalb分类预测)GA-BP遗传算法优化BP神经网络的多维分类预测
  • (SpringBoot)第二章:Spring创建和使用
  • (zt)最盛行的警世狂言(爆笑)
  • (大众金融)SQL server面试题(1)-总销售量最少的3个型号的车及其总销售量
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理 第13章 项目资源管理(七)
  • (篇九)MySQL常用内置函数
  • (原创)攻击方式学习之(4) - 拒绝服务(DOS/DDOS/DRDOS)
  • **《Linux/Unix系统编程手册》读书笔记24章**
  • ..回顾17,展望18
  • .NET CORE Aws S3 使用
  • .net oracle 连接超时_Mysql连接数据库异常汇总【必收藏】
  • .net 调用php,php 调用.net com组件 --
  • .NET 读取 JSON格式的数据
  • .net 中viewstate的原理和使用