数据结构与算法系列——栈

什么是栈

栈是一种运算受限制的线性表,只允许在表的一端进行插入和删除操作。这一端被称为栈顶,另一端被称为栈底。向一个栈中插入新数据叫做进栈、入栈或者压栈,是把新元素放到栈顶上边,使其成为新的栈顶元素;删除数据叫做出栈或者退栈,就是把栈顶的元素删掉,使其下边的元素称为新的栈顶元素。

举一个容易理解的例子,就是有一摞盘子,我们用的时候从上往下一个一个取,放的时候都是从下往上一个一个放,一般不从中间取或者放。这种先进后出,后进先出的数据结构就是栈。

这种操作受限的数据结构在什么情况下用呢,我们为什么不能用操作更为方便的数组或者链表呢。当某个数据只涉及在一端的插入和删除数据,并满足先进后出,后进先出的特点,我们就可以用栈这种数据结构,而数组和链表因为操作的灵活性,有时候会使一些数据不可控,更容易出现错误。

栈的实现

从功能上,我们是可以用数组和链表来实现栈,只要实现栈的入栈和出栈的操作,即从栈顶插入新的数据,从栈顶删除数据。用数组实现的栈叫做顺序栈,用链表实现的栈叫做链表栈。下边我们分别看一下用数组和链表实现栈的代码。这里用Java代码实现。

  • 数组实现
//基于数组实现的栈
public class ArrayStack{

    //数组
    private String[] items;
    //栈的大小
    private int length;
    //栈中元素的个数
    private int count;


    public ArrayStack(int len){
        items = new String[len];
        length = len;
        count = 0;
    }

    //入栈
    public boolean Push(String x){
        //数组空间不足
        if(count == length){
            return false;
        }
        items[count] = x;
        count++;
        return true;
    }

    //出栈
    public String Pop(){
        //栈为空
        if(count == 0){
            return null;
        }
        String tem = items[count-1];
        count--;
        return tem;
    }

}

  • 链表实现
//基于链表实现的栈
public class ListNodeStack {

    private ListNode top;

    //进栈
    public void Push(int val) {
        ListNode node = new ListNode(val, null);
        if (top == null) {
            top = node;
        } else {
            node.next = top;
            top = node;
        }
    }

    //出栈
    public int Pop() {
        if (top == null) {
            return -1;
        }
        int val = top.val;
        top = top.next;
        return val;
    }


    //链表的结点
    class ListNode {
        private int val;
        private ListNode next;

        public ListNode(int x, ListNode next) {
            val = x;
            this.next = next;
        }

        public int GetValue() {
            return val;
        }
    }
}

上边代码用数组实现的栈是一个固定大小的栈,当栈满了之后就没法办插入新的数据了,那么我们能不能用数组实现一个动态扩容的栈呢?前边我们将数组的时候说过,实现一个动态扩容的数组,是在数组满了的时候,我们重新创建一个大小为原来两倍的数组,然后把原来数组的数据拷贝到新的数组中,所以我们也可以用这个方法来实现一个动态扩容的栈。我们看一下代码实现。

//基于数组实现的栈
public class ArrayStack {

    //数组
    private String[] items;
    //栈的大小
    private int length;
    //栈中元素的个数
    private int count;


    public ArrayStack(int len) {
        items = new String[len];
        length = len;
        count = 0;
    }

    //入栈
    public void Push(String x) {
        //数组空间不足
        if (count == length) {
            DilatationArray();
        }
        items[count] = x;
        count++;
    }

    //出栈
    public String Pop() {
        //栈为空
        if (count == 0) {
            return null;
        }
        String tem = items[count - 1];
        count--;
        return tem;
    }

    //数组扩容
    private void DilatationArray() {
        String[] newArray = new String[length * 2];
        for (int i = 0; i < length; i++) {
            newArray[i] = items[I];
        }
        items = newArray;
    }

}

栈的实际应用
  1. 在函数调用中的应用

在Java的虚拟机中有一个内存区域被称为虚拟机栈。每个方法在执行的时候都会创建一个“栈帧”。用来存储局部变量表(包括参数)、操作栈、动态链接、方法出口等信息。每个方法从调用到结束就会有栈帧在虚拟机栈中入栈和出栈。

举一个简单的例子。

public class AddClass{
    public int Main(){
        int a = 0;
        int b = 5;
        int c = 0;
        a = Add(2, 3);
        c = a + b;
        return c;
    }
    
    public int Add(int x, int y){
        int sum = 0;
        sum = x + y;
        return sum;
    }
}

从代码中我们看到 Main 方法中首先声明了几个变量,然后调用了 Add 方法,然后经过一些运算,最后返回一个值。我们画图来更直观的看一下这个过程。

image.png

  1. 在表达式求值中的应用

编辑器的表达式求值的过程,就是用栈来实现的。我们举一个简单的四则运算的表达式的求值过程来看一下。例如:1+2*3-4/2。编辑器是怎么计算来得到最后的值呢。

这个求值过程,编辑器是用两个栈来实现的,一个保存数字的栈,一个保存运算符号的栈。我们从左向右遍历表达式,当遇到数字的时候把它压入数字栈,当遇到运算符号的时候,就与运算符栈的栈顶的运算符比较,如果比栈顶的运算符优先级高,就直接压入运算符栈,如果比栈顶的运算符优先级低或者相同,那么就从运算符栈取出栈顶运算符号,从数字栈中取出两个数字进行计算,然后把结果压入数字栈,然后继续比较,依次类推,知道最后。

我们为了更形象的理解,也用画图的方式来展示一下这个过程。

image.png

  1. 在各种前进后退操作中的应用

我们在各中编辑器的撤销和恢复操作,浏览器中的前进和后退操作,都是用栈来实现的。

我们用两个栈 A 和 B,当我们执行操作的时候把我们的每一个操作依次压入 A 栈中,当我们执行后退的操作时,依次从 A 栈中取出,然后压入 B 栈中,当执行前进操作的时候,依次从 B 栈中取出,然后压入 A 栈中。

比如我们依次执行了 a,b,c 操作,我们依次把 a,b,c 压入 A 栈。如图

image.png

假如我们现在想要撤销 c 和 b 操作。那么它就是这样的。

image.png

假如我又想恢复操作 b
image.png

这个时候我继续执行新的操作 d,那么无论前进后退我们都无法再回到操作 c 了,所以我们应该清空 B 栈。
image.png

欢迎关注公众号:「努力给自己看」

公众号200x200

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值