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

了解栈Stack一篇文章就够了

  1. 什么是栈

栈是一种特殊的线性表,只允许一端进行数据的插入和删除,即先进后出原则。类似于弹夹先装进去的子弹后面出,后放入的子弹先出。

  1. 栈的底层原理

栈是一种线性结构,所以既能使用数组实现,也能使用链表实现,我就采用数组的方法实现一下栈
public class MyStack {public int[] elem;private static final int DEFAULT_CAPACITY = 10;public MyStack() {elem = new int[DEFAULT_CAPACITY];}//既能统计栈中存储多少个有效数据,也可以作为存放元素的下标public int usedSize;//压栈public void push(int val) {//先判满,如果满了要扩容if(isFull()) {elem = Arrays.copyOf(elem, elem.length * 2);}elem[usedSize] = val;usedSize++;}private boolean isFull() {return usedSize == elem.length;}//出栈public int pop() {//先判空,如果栈为空抛出异常if(empty()) {throw new EmptyArrayException("栈为空");}int val = elem[usedSize-1];usedSize--;return val;}private boolean empty() {return usedSize == 0;}//查看栈顶元素public int peek() {if(empty()) {throw new EmptyArrayException("栈为空");}return elem[usedSize-1];}//获取栈中有多少元素public int getUsedSize() {return usedSize;}
}
使用数组的实现方式,入栈只需要elem[usedSize] = data和出栈只需要usedSize--的时间复杂度都是O(1)。
如果使用单向链表,采用头插法入栈和出栈,只需要移动head头节点,时间复杂度也是O(1);要是采用尾插法入栈和出栈,入栈和出栈都需要从头节点出发,走到尾巴,才能完成元素的入栈或出栈,时间复杂度为O(n).
如果是双向链表,那么一个节点即知道后继节点,也知道前驱节点,有头节点head也有尾节点tail,采用头插法进行入栈和出栈只需要移动head头节点,采用尾插法进行入栈和出栈也是一样,只需要移动tail尾巴节点。
  1. 栈的可能出栈顺序

题目:若入栈序列为1,2,3,4,进栈的过程中可以出栈,则下面不可能的出栈序列是哪个?

A.1,4,3,2 B.2,3,4,1 C.3,1,4,2 D.3,4,2,1
方法:按题目给的入栈顺序,结合选项,加画图(熟练以后不用画图)
A选项:第一个出栈是1,所以1进栈后出栈,2进栈,3进栈(因为出栈的第二个是4,所以2,3没有出栈),4进栈。4出栈,3出栈,2出栈。所以A是可能的出栈序列
B选项:B选项第一个出栈的是2,所以1进栈2进栈,2出栈,3进栈3出栈,4进栈4出栈,此时栈还剩1,1出栈。也是可能的出栈序列
C选项:你看C选项第一个出栈的是3,所以1,2,3都进栈,3再出栈,它第二个出栈的是1,但是此时栈顶元素是2,所以这是不可能的出栈顺序
D选项:第一个出栈的是3,所以要1,2,3都入栈了,3是栈顶元素才能出栈,接着4入栈,4又出栈,然后把栈中2出栈,1出栈

有关出栈顺序的oj题: https://www.nowcoder.com/practice/d77d11405cc7470d82554cb392585106?tpId=13&&tqId=11174&rp=1&ru=/activity/oj&qru=/ta/coding-interviews/question-ranking

public class Solution {public boolean IsPopOrder(int [] pushA, int [] popA) {Stack<Integer> stack = new Stack<>();int index = 0;//popA数组下标//遍历压栈pushA数组for (int i = 0; i < pushA.length; i++) {stack.push(pushA[i]);//如果栈顶元素和popA[index]相等//并且要防止栈为空(空指针异常)和数组越界//使用循环可能连着多个相等while (!stack.empty() && index < popA.length && stack.peek() == popA[index]) {stack.pop();index++;}}//如果是可能的出栈序列,那么栈中的元素能全部匹配,全部被弹出//如果栈不为空,说明是不可能的出栈序列return stack.empty();}
}
  1. 中缀表达式转后缀表达式

题目1:中缀表达式:2+3*6的后缀表达式?
题目2 :中缀表达式:(2+3)*5的后缀表达式
中缀转后缀,我们是加括号,然后把操作符移到对应括号后面,再去括号。同理,中缀转前缀,就是把操作符移到括号前面
  1. 求后缀表达式/逆波兰表达式的计算结果

力扣Oj: https://leetcode.cn/problems/evaluate-reverse-polish-notation/

解决方法:使用栈,将数字放入栈中,如果碰到操作符,取出栈顶的2个元素,第一个取出的放操作符右边,第二个取出放操作符左边(固定的),计算完了再放入栈中,直到遍历完该字符串数组,此时栈中元素的值就是结果
class Solution {public int evalRPN(String[] tokens) {//等会要进行运算,类型用IntegerStack<Integer> stack = new Stack<>();//遍历数组for(String x: tokens) {//判断是不是数字字符串if(! isOperation(x)) {//如果是数字字符串,入栈stack.push(Integer.parseInt(x));//转为数字}else {//碰到操作符,出栈2个数int num1 = stack.pop();int num2 = stack.pop();//根据不同操作字符,进行运算,第一个弹出放操作符右边,第二个放左边//运算完后,重新入栈switch(x) {case "+" :stack.push(num2 + num1);break;case "-" :stack.push(num2 - num1);break;case "*":stack.push(num2 * num1);break;case "/" :stack.push(num2 / num1);break;} }}//此时栈中元素就是运算结果return stack.pop();}private boolean isOperation(String x) {//字符串比较不能使用==,==是比较地址,而不是比较值//字符串重写了Object类的equals()方法,是比较值的if(x.equals("+") || x.equals("-") || x.equals("*") || x.equals("/")) {return true;//如果是操作符返回真}return false;//不是返回假}
}

http://www.lryc.cn/news/34380.html

相关文章:

  • CNStack 助推龙源电力扛起“双碳”大旗
  • ruoyi-vue-plus1(控制台相关的输出日志)(p6spy插件)(jackson全局配置)(StopWatch)
  • 【Mybatis】| 如何创建MyBatis的工具类
  • 【Java】DT怎么写?
  • xcode14安装swift package设置github账户token
  • css面试题1
  • Hive基础
  • 信息收集-
  • 【sdx12】sdx12获取Serial Number操作方法及源码分享Serial Number的寄存器地址
  • 23种设计模式-工厂模式(安卓应用场景介绍)
  • sheng的学习笔记-服务熔断与降级组件Hystrix
  • 简单给WordPress怎么添加自定义字段面板
  • 大数据框架之Hive:第6章 查询
  • CentOS 8搭建EMQX集群
  • 基于神经网络的自监督学习方法音频分离器(Matlab代码实现)
  • yocto 如何添加python module
  • [深入理解SSD系列综述 2.1.2] SLC、MLC、TLC、QLC、PLC NAND_固态硬盘闪存颗粒类型
  • Matlab实现FFT变换
  • JVM调优面试题——垃圾回收专题
  • java启动命令中-D和--的区别
  • QML Popup详解
  • [2.1.6]进程管理——线程的实现方式和多线程模型
  • 小白做什么兼职项目赚钱?宝妈拍短视频赚钱的方法
  • 第十四届蓝桥杯第三期模拟赛 C/C++ B组 原题与详解
  • Linux中断操作
  • 看看CabloyJS是如何异步加载并执行go wasm模块的
  • 嵌入式C语言九大数据结构操作方式详解
  • 【C++学习】栈 | 队列 | 优先级队列 | 反向迭代器
  • Python—看我分析下已经退市的 可转债 都有什么特点
  • 【第八课】空间数据基础与处理——数据结构转化