1. 堆栈简介

堆栈(Stack):简称为栈。一种线性表数据结构,是一种只允许在表的一端进行插入和删除操作的线性表。

我们把栈中允许插入和删除的一端称为 「栈顶(top)」;另一端则称为 「栈底(bottom)」。当表中没有任何数据元素时,称之为 「空栈」

堆栈有两种基本操作:「插入操作」「删除操作」

  • 栈的插入操作又称为「入栈」或者「进栈」。
  • 栈的删除操作又称为「出栈」或者「退栈」。

img

简单来说,栈是一种 「后进先出(Last In First Out)」 的线性表,简称为 「LIFO 结构」

我们可以从两个方面来解释一下栈的定义:

  • 第一个方面是 「线性表」

栈首先是一个线性表,栈中元素具有前驱后继的线性关系。栈中元素按照 a1,a2,…,an 的次序依次进栈。栈顶元素为 an

  • 第二个方面是 「后进先出原则」

根据堆栈的定义,每次删除的总是堆栈中当前的栈顶元素,即最后进入堆栈的元素。而在进栈时,最先进入堆栈的元素一定在栈底,最后进入堆栈的元素一定在栈顶。也就是说,元素进入堆栈或者退出退栈是按照「后进先出(Last In First Out)」的原则进行的。

2. 堆栈的顺序存储与链式存储

和线性表类似,栈有两种存储表示方法:「顺序栈」「链式栈」

  • 「顺序栈」:即堆栈的顺序存储结构。利用一组地址连续的存储单元依次存放自栈底到栈顶的元素,同时使用指针 top 指示栈顶元素在顺序栈中的位置。
  • 「链式栈」:即堆栈的链式存储结构。利用单链表的方式来实现堆栈。栈中元素按照插入顺序依次插入到链表的第一个节点之前,并使用栈顶指针 top 指示栈顶元素,top 永远指向链表的头节点位置。

在描述堆栈的顺序存储与链式存储具体实现之前,我们先来看看堆栈具有哪些基本操作。

2.1 堆栈的基本操作

栈作为一种线性表来说,理论上应该具备线性表所有的操作特性,但由于「后进先出」的特殊性,所以针对栈的操作进行了一些变化。尤其是插入操作和删除操作,改为了入栈(push)和出栈(pop)。

堆栈的基本操作如下:

  • 初始化空栈:创建一个空栈,定义栈的大小 size,以及栈顶元素指针 top
  • 判断栈是否为空:当堆栈为空时,返回 True。当堆栈不为空时,返回 False。一般只用于栈中删除操作和获取当前栈顶元素操作中。
  • 判断栈是否已满:当堆栈已满时,返回 True,当堆栈未满时,返回 False。一般只用于顺序栈中插入元素和获取当前栈顶元素操作中。
  • 插入元素(进栈、入栈):相当于在线性表最后元素后面插入一个新的数据元素。并改变栈顶指针 top 的指向位置。
  • 删除元素(出栈、退栈):相当于在线性表最后元素后面删除最后一个数据元素。并改变栈顶指针 top 的指向位置。
  • 获取栈顶元素:相当于获取线性表中最后一个数据元素。与插入元素、删除元素不同的是,该操作并不改变栈顶指针 top 的指向位置。

接下来我们来看一下栈的顺序存储与链式存储两种不同的实现方式。

2.2 堆栈的顺序存储实现

堆栈最简单的实现方式就是借助于一个数组来描述堆栈的顺序存储结构。在 Python 中我们可以借助列表 list 来实现。这种采用顺序存储结构的堆栈也被称为 「顺序栈」

2.2.1 堆栈的顺序存储基本描述

img

我们约定 self.top 指向栈顶元素所在位置。

  • 初始化空栈:使用列表创建一个空栈,定义栈的大小 self.size,并令栈顶元素指针 self.top 指向 -1,即 self.top = -1
  • 判断栈是否为空:当 self.top == -1 时,说明堆栈为空,返回 True,否则返回 False
  • 判断栈是否已满:当 self.top == self.size - 1,说明堆栈已满,返回 True,否则返回返回 False
  • 插入元素(进栈、入栈):先判断堆栈是否已满,已满直接抛出异常。如果堆栈未满,则在 self.stack 末尾插入新的数据元素,并令 self.top 向右移动 1 位。
  • 删除元素(出栈、退栈):先判断堆栈是否为空,为空直接抛出异常。如果堆栈不为空,则删除 self.stack 末尾的数据元素,并令 self.top 向左移动 1 位。
  • 获取栈顶元素:先判断堆栈是否为空,为空直接抛出异常。不为空则返回 self.top 指向的栈顶元素,即 self.stack[self.top]

2.2.2 堆栈的顺序存储实现代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
public class Stack {
private List<Integer> stack;
private int size;
private int top;

// 初始化空栈
public Stack(int size) {
this.stack = new ArrayList<>(size);
this.size = size;
this.top = -1;
}

// 默认大小为100的构造函数
public Stack() {
this(100);
}

// 判断栈是否为空
public boolean isEmpty() {
return top == -1;
}

// 判断栈是否已满
public boolean isFull() {
return top + 1 == size;
}

// 入栈操作
public void push(int value) {
if (isFull()) {
throw new RuntimeException("Stack is full");
} else {
stack.add(value);
top++;
}
}

// 出栈操作
public void pop() {
if (isEmpty()) {
throw new EmptyStackException();
} else {
stack.remove(top);
top--;
}
}

// 获取栈顶元素
public int peek() {
if (isEmpty()) {
throw new EmptyStackException();
} else {
return stack.get(top);
}
}
}

2.3 堆栈的链式存储实现

堆栈的顺序存储结构保留着顺序存储分配空间的固有缺陷,即在栈满或者其他需要重新调整存储空间时需要移动大量元素。为此,堆栈可以采用链式存储方式来实现。在 Python 中我们通过构造链表节点 Node 的方式来实现。这种采用链式存储结构的堆栈也被称为 「链式栈」

img

2.3.1 堆栈的链式存储基本描述

我们约定 self.top 指向栈顶元素所在位置。

  • 初始化空栈:使用列表创建一个空栈,并令栈顶元素指针 self.top 指向 None,即 self.top = None
  • 判断栈是否为空:当 self.top == None 时,说明堆栈为空,返回 True,否则返回 False
  • 插入元素(进栈、入栈):创建值为 value 的链表节点,插入到链表头节点之前,并令栈顶指针 self.top 指向新的头节点。
  • 删除元素(出栈、退栈):先判断堆栈是否为空,为空直接抛出异常。如果堆栈不为空,则先使用变量 cur 存储当前栈顶指针 self.top 指向的头节点,然后令 self.top 沿着链表移动 1 位,然后再删除之前保存的 cur 节点。
  • 获取栈顶元素:先判断堆栈是否为空,为空直接抛出异常。不为空则返回 self.top 指向的栈顶节点的值,即 self.top.value

2.3.2 堆栈的链式存储实现代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
public class Node {
public int value;
public Node next;

public Node(int value) {
this.value = value;
this.next = null;
}
}

public class Stack {
private Node top;

// 初始化空栈
public Stack() {
this.top = null;
}

// 判断栈是否为空
public boolean isEmpty() {
return top == null;
}

// 入栈操作
public void push(int value) {
Node newNode = new Node(value);
newNode.next = top;
top = newNode;
}

// 出栈操作
public void pop() {
if (isEmpty()) {
throw new RuntimeException("Stack is empty");
} else {
Node temp = top;
top = top.next;
temp = null; // 将原栈顶节点释放,可以由垃圾收集器回收
}
}

// 获取栈顶元素
public int peek() {
if (isEmpty()) {
throw new RuntimeException("Stack is empty");
} else {
return top.value;
}
}
}

3. 堆栈的应用

堆栈是算法和程序中最常用的辅助结构,其的应用十分广泛。堆栈基本应用于两个方面:

  • 使用堆栈可以很方便的保存和取用信息,因此长被用作算法和程序中的辅助存储结构,临时保存信息,供后面操作中使用。
    • 例如:操作系统中的函数调用栈,浏览器中的前进、后退功能。
  • 堆栈的后进先出规则,可以保证特定的存取顺序。
    • 例如:翻转一组元素的顺序、铁路列车车辆调度。

下面我们来讲解一下栈应用的典型例子。

3.1 括号匹配问题

3.1.1 题目链接

3.1.2 题目大意

描述:给定一个只包括 '('')''{''}''['']' 的字符串 s

要求:判断字符串 s 是否有效(即括号是否匹配)。

说明

  • 有效字符串需满足:
    1. 左括号必须用相同类型的右括号闭合。
    2. 左括号必须以正确的顺序闭合。

示例

1
2
3
4
5
6
输入:s = "()"
输出:True


输入:s = "()[]{}"
输出:True

3.2.3 解题思路

思路 1:栈

括号匹配是「栈」的经典应用。我们可以用栈来解决这道题。具体做法如下:

  1. 先判断一下字符串的长度是否为偶数。因为括号是成对出现的,所以字符串的长度应为偶数,可以直接判断长度为奇数的字符串不匹配。如果字符串长度为奇数,则说明字符串 s 中的括号不匹配,直接返回 False
  2. 使用栈stack来保存未匹配的左括号。然后依次遍历字符串s中的每一个字符。
    1. 如果遍历到左括号时,将其入栈。
    2. 如果遍历到右括号时,先看栈顶元素是否是与当前右括号相同类型的左括号。
      1. 如果是与当前右括号相同类型的左括号,则令其出栈,继续向前遍历。
      2. 如果不是与当前右括号相同类型的左括号,则说明字符串 s 中的括号不匹配,直接返回 False
  3. 遍历完,还要再判断一下栈是否为空。
    1. 如果栈为空,则说明字符串 s 中的括号匹配,返回 True
    2. 如果栈不为空,则说明字符串 s 中的括号不匹配,返回 False
思路 1:代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
public class Solution {
public boolean isValid(String s) {
if (s.length() % 2 == 1) {
return false;
}

Stack<Character> stack = new Stack<>();

for (char ch : s.toCharArray()) {
if (ch == '(' || ch == '[' || ch == '{') {
stack.push(ch);
} else if (ch == ')') {
if (!stack.isEmpty() && stack.peek() == '(') {
stack.pop();
} else {
return false;
}
} else if (ch == ']') {
if (!stack.isEmpty() && stack.peek() == '[') {
stack.pop();
} else {
return false;
}
} else if (ch == '}') {
if (!stack.isEmpty() && stack.peek() == '{') {
stack.pop();
} else {
return false;
}
}
}

return stack.isEmpty();
}
}

思路 1:复杂度分析
  • 时间复杂度:O(n)。
  • 空间复杂度:O(1)。

3.2 表达式求值问题

3.2.1 题目链接

3.2.2 题目大意

描述:给定一个字符串表达式 s,表达式中所有整数为非负整数,运算符只有 +-*/,没有括号。

要求:实现一个基本计算器来计算并返回它的值。

说明

  • 1≤s.length≤3∗105
  • s 由整数和算符(+-*/)组成,中间由一些空格隔开。
  • s 表示一个有效表达式。
  • 表达式中的所有整数都是非负整数,且在范围 [0,231−1] 内。
  • 题目数据保证答案是一个 32-bit 整数。

示例

1
2
3
4
5
6
输入:s = "3+2*2"
输出:7


输入:s = " 3/2 "
输出:1

3.2.3 解题思路

思路 1:栈

计算表达式中,乘除运算优先于加减运算。我们可以先进行乘除运算,再将进行乘除运算后的整数值放入原表达式中相应位置,再依次计算加减。

可以考虑使用一个栈来保存进行乘除运算后的整数值。正整数直接压入栈中,负整数,则将对应整数取负号,再压入栈中。这样最终计算结果就是栈中所有元素的和。

具体做法:

  1. 遍历字符串 s,使用变量 op 来标记数字之前的运算符,默认为 +
  2. 如果遇到数字,继续向后遍历,将数字进行累积,得到完整的整数 num。判断当前 op 的符号。
    1. 如果 op+,则将 num 压入栈中。
    2. 如果 op-,则将 -num 压入栈中。
    3. 如果 op*,则将栈顶元素 top 取出,计算 top * num,并将计算结果压入栈中。
    4. 如果 op/,则将栈顶元素 top 取出,计算 int(top / num),并将计算结果压入栈中。
  3. 如果遇到 +-*/ 操作符,则更新 op
  4. 最后将栈中整数进行累加,并返回结果。
思路 1:代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
public class Solution {
public int calculate(String s) {
int size = s.length();
Stack<Integer> stack = new Stack<>();
char op = '+';
int num = 0;
int index = 0;

while (index < size) {
if (s.charAt(index) == ' ') {
index++;
continue;
}

if (Character.isDigit(s.charAt(index))) {
num = s.charAt(index) - '0';

while (index + 1 < size && Character.isDigit(s.charAt(index + 1))) {
index++;
num = 10 * num + s.charAt(index) - '0';
}

if (op == '+') {
stack.push(num);
} else if (op == '-') {
stack.push(-num);
} else if (op == '*') {
int top = stack.pop();
stack.push(top * num);
} else if (op == '/') {
int top = stack.pop();
stack.push(top / num);
}
} else if (s.charAt(index) == '+' || s.charAt(index) == '-' || s.charAt(index) == '*' || s.charAt(index) == '/') {
op = s.charAt(index);
}

index++;
}

int result = 0;
while (!stack.isEmpty()) {
result += stack.pop();
}

return result;
}
}
思路 1:复杂度分析
  • 时间复杂度:O(n)。
  • 空间复杂度:O(n)。

本站由 卡卡龙 使用 Stellar 1.29.1主题创建

本站访问量 次. 本文阅读量 次.