5.1 栈的一个实际需求

请输入一个表达式
计算式:[722-5+1-5+3-3] 点击计算【如下图】
image.png
请问: 计算机底层是如何运算得到结果的?
注意不是简单的把算式列出运算,因为我们看这个算式7 2 2 -5, 但是计算机怎么理解这个算式的(对计算机而言,它接收到的就是一个字符串),我们讨论的是这个问题。-> 栈

5.2 栈的介绍

1) 栈的英文为(stack)
2) 栈是一个先入后出(FILO-First In Last Out)的有序列表。
3) 栈(stack)是限制线性表中元素的插入和删除只能在线性表的同一端进行的一种特殊线性表。允许插入和删除的
一端,为变化的一端,称为栈顶(Top),另一端为固定的一端,称为栈底(Bottom)
4) 根据栈的定义可知,最先放入栈中元素在栈底,最后放入的元素在栈顶,而删除元素刚好相反,最后放入的元
素最先删除,最先放入的元素最后删除
5) 图解方式说明出栈(pop)和入栈(push)的概念
image.png
image.png

5.3 栈的应用场景

1) 子程序的调用:在跳往子程序前,会先将下个指令的地址存到堆栈中,直到子程序执行完后再将地址取出,以
回到原来的程序中。类似return
2) 处理递归调用:和子程序的调用类似,只是除了储存下一个指令的地址外,也将参数、区域变量等数据存入堆
栈中。
3) 表达式的转换[中缀表达式转后缀表达式]与求值(实际解决)。
4) 二叉树的遍历
5) 图形的深度优先(depth 一first)搜索法

5.4 栈的快速入门

5.4.1 用数组模拟栈的使用

由于栈是一种有序列表,当然可以使用数组的结构来储存栈的数据内容,
下面我们就用数组模拟栈的出栈,入栈等操作。

5.4.2 数组模拟栈的思路分析图

image.png
实现栈的思路分析
1. 使用数组来模拟栈
2. 定义一个 top 来表示栈顶,初始化 为 -1
3. 入栈的操作,当有数据加入到栈时, top++; stack[top] = data;
4. 出栈的操作, int value = stack[top]; top—, return value

5.4.3 代码实现

  1. package stack;
  2. import java.util.Scanner;
  3. public class ArrayStackDemo {
  4. public static void main(String[] args) {
  5. //测试一下ArrayStack 是否正确
  6. //先创建一个ArrayStack 对象->表示栈
  7. ArrayStack stack = new ArrayStack(4);
  8. String key = "";
  9. boolean loop = true; //控制是否退出菜单
  10. Scanner scanner = new Scanner(System.in);
  11. while(loop) {
  12. System.out.println("show: 表示显示栈");
  13. System.out.println("exit: 退出程序");
  14. System.out.println("push: 表示添加数据到栈(入栈)");
  15. System.out.println("pop: 表示从栈取出数据(出栈)");
  16. System.out.println("请输入你的选择");
  17. key = scanner.next();
  18. switch (key) {
  19. case "show":
  20. stack.list();
  21. break;
  22. case "push":
  23. System.out.println("请输入一个数");
  24. int value = scanner.nextInt();
  25. stack.push(value);
  26. break;
  27. case "pop":
  28. try {
  29. int res = stack.pop();
  30. System.out.printf("出栈的数据是%d\n", res);
  31. } catch (Exception e) {
  32. // TODO: handle exception
  33. System.out.println(e.getMessage());
  34. }
  35. break;
  36. case "exit":
  37. scanner.close();
  38. loop = false;
  39. break;
  40. default:
  41. break;
  42. }
  43. }
  44. System.out.println("程序退出~~~");
  45. }
  46. }
  47. //定义一个ArrayStack 表示栈
  48. class ArrayStack {
  49. private int maxSize; // 栈的大小
  50. private int[] stack; // 数组,数组模拟栈,数据就放在该数组
  51. private int top = -1;// top 表示栈顶,初始化为-1
  52. //构造器
  53. public ArrayStack(int maxSize) {
  54. this.maxSize = maxSize;
  55. stack = new int[this.maxSize];
  56. }
  57. //栈满
  58. public boolean isFull() {
  59. return top == maxSize - 1;
  60. }
  61. //栈空
  62. public boolean isEmpty() {
  63. return top == -1;
  64. }
  65. //入栈-push
  66. public void push(int value) {
  67. //先判断栈是否满
  68. if(isFull()) {
  69. System.out.println("栈满");
  70. return;
  71. }
  72. top++;
  73. stack[top] = value;
  74. }
  75. //出栈-pop, 将栈顶的数据返回
  76. public int pop() {
  77. //先判断栈是否空
  78. if(isEmpty()) {
  79. //抛出异常
  80. throw new RuntimeException("栈空,没有数据~");
  81. }
  82. int value = stack[top];
  83. top--;
  84. return value;
  85. }
  86. //显示栈的情况[遍历栈], 遍历时,需要从栈顶开始显示数据
  87. public void list() {
  88. if(isEmpty()) {
  89. System.out.println("栈空,没有数据~~");
  90. return;
  91. }
  92. //需要从栈顶开始显示数据
  93. for(int i = top; i >= 0 ; i--) {
  94. System.out.printf("stack[%d]=%d\n", i, stack[i]);
  95. }
  96. }
  97. }