目录
基于数组和链表实现栈
1 栈的基础
栈和队列是比较特殊的线性表,又称为访问受限的线性表。栈是很多表达式、符号等运算的基础,也是递归的底层实现。栈底层实现任然是链表或者顺序表,栈与线性表最大区别是数据的存取被限制了,其插入和删除只允许在线性表的一段进行。一般而言,允许操作的一段称为栈顶,不可操作的一端称为栈底,插入元素的操作称为入栈,删除元素的操作称为出栈。若栈中无任何元素,称为空栈。
栈的常用操作主要有
1)push(E):增加一个元素E
2)pop():弹出元素E
3) peek(): 显示栈顶元素,但是不出栈
4)empty(): 判断栈是否为空
java的util中提供了栈Stack类,下面是一个实现的例子:
public class MainTest{
public static void main(String[] args){
Stack<Integer> stack = new Stack(); // 构建栈
stack.push(1); // 入栈
stack.push(2);
stack.push(3);
System.out.println("栈顶元素:"+ stack.peek()); // 查看栈顶元素
while(!stack.empty()){
System.out.println("栈顶元素:"+ stack.peek()); // 只显示没出栈
System.out.println("栈顶元素:"+ stack.pop()); // 出栈并且显示
}
}
}
2 基于数组实现栈
采用顺序表实现的栈,内部以数组为基础,实现对元素的存取操作。应用中要注意每次入栈之前先判断栈的容量是否够用,如果不够用可以进行扩容。
入栈出栈过程如下所示:
出栈时 top先将栈顶元素取出,然后再执行top--。实现的完整代码如下:
import java.util.Arrays;
class Mystack<T> {
// 实现栈的数组
private Object[] stack;
// 栈顶元素
private int top;
Mystack() {
// 栈初始容量为10
stack = new Object[10];
}
// 判断栈是否为空
public boolean isEmpty() {
return top == 0;
}
// 返回栈顶元素,只返回不出栈
public T peek() {
T t = null;
if (top > 0)
t = (T) stack[top - 1];
return t;
}
// 入栈操作,入栈后top++
public void push(T t) {
expandCapacity(top + 1);
stack[top] = t;
top++;
}
//出栈操作,出栈后top--
public T pop() {
T t = peek();
if (top > 0) {
stack[top - 1] = null;
top--;
}
return t;
}
//扩大容量,扩大为原来的1.5倍
public void expandCapacity(int size) {
int len = stack.length;
if (size > len) {
size = size * 3 / 2 + 1;//每次扩大50%
stack = Arrays.copyOf(stack, size);
}
}
// 测试类
public static void main(String[] args) {
Mystack<String> stack = new Mystack<>();
System.out.println(stack.peek());
System.out.println(stack.isEmpty());
stack.push("java");
stack.push("is");
stack.push("beautiful");
stack.push("language");
System.out.println(stack.pop());
System.out.println(stack.isEmpty());
System.out.println(stack.peek());
}
}
3 基于链表实现栈
链表也可以实现栈,插入和删除都在头结点进行就可以了。在此之前我们已经学习过没有虚拟节点时对链表头元素进行插入和删除了,而这里基于链表实现栈是一模一样的
完整代码实现如下:
class ListStack<T> {
//定义链表节点
class Node<T> {
public T t;
public Node next;
}
// 构造头结点
public Node<T> head;
// 初始化函数初始化头指针
ListStack() {
head = null;
}
// 入栈
public void push(T t) {
if (t == null) {
throw new NullPointerException("参数不能为空");
}
// 如果为空栈,那么直接作为头结点并指向空
if (head == null) {
head = new Node<T>();
head.t = t;
head.next = null;
} else {
// 在链表表头插入
Node<T> temp = head;
head = new Node<>();
head.t = t;
head.next = temp;
}
}
//出栈
public T pop() {
// 空栈直接返回空
if (head == null) {
return null;
}
// 头结点变为下一个,并返回出栈的头结点值
T t = head.t;
head = head.next;
return t;
}
//取栈顶元素
public T peek() {
if (head == null) {
return null;
}
T t = head.t;
return t;
}
// 判断栈空
public boolean isEmpty() {
if (head == null)
return true;
else
return false;
}
// 测试类
public static void main(String[] args) {
ListStack stack = new ListStack();
System.out.println(stack.isEmpty());
stack.push("I");
stack.push("am");
stack.push("crazy");
System.out.println(stack.peek());
System.out.println(stack.pop());
System.out.println(stack.isEmpty());
}
}