【数据结构】栈(基于数组、链表实现 + GIF图解 + 原码)

Hi~!这里是奋斗的明志,很荣幸您能阅读我的文章,诚请评论指点,欢迎欢迎 ~~
🌱🌱个人主页:奋斗的明志
🌱🌱所属专栏:数据结构

在这里插入图片描述

📚本系列文章为个人学习笔记,在这里撰写成文一为巩固知识,二为展示我的学习过程及理解。文笔、排版拙劣,望见谅。

在这里插入图片描述


前言

一、栈(Stack)

1.概念

  • :一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈 顶,另一端称为栈底。栈中的数据元素遵守后进先出LIFO( Last In First Out)的原则。
  • 压栈:栈的插入操作叫做进栈/压栈/入栈,入数据在栈顶
  • 出栈:栈的删除操作叫做出栈。 出数据在栈顶
  • 栈顶栈底: 这个描述是偏向于逻辑上的内容,因为大家知道数组在末尾插入删除更容易,而单链表通常在头插入删除更容易。所以数组可以用末尾做栈顶,而链表可以头做栈顶

2.栈在现实生活中的例子

在这里插入图片描述

在这里插入图片描述

栈的应用广泛,比如你的程序执行查看调用堆栈、计算机四则加减运算、算法的非递归形式、括号匹配问题等等。所以栈也是必须掌握的一门数据结构。最简单大家都经历过,你拿一本书上下叠在一起,就是一个后进先出的过程,你可以把它看成一个栈。下面我们介绍数组实现的栈链表实现的栈

二、栈的使用

1.方法

方法功能
Stack()构造一个空的栈
E push(E e)将e入栈,并返回e
E pop()将栈顶元素出栈并返回
E peek()获取栈顶元素
int size()获取栈中有效元素个数
boolean empty()检测栈是否为空

2.代码

代码如下(示例):

public static void main(String[] args) {
    Stack<Integer> s = new Stack();
    s.push(1);
    s.push(2);
    s.push(3);
    s.push(4);
    System.out.println(s.size());  // 获取栈中有效元素个数---> 4 System.out.println(s.peek());  // 获取栈顶元素---> 4
    s.pop();  // 4出栈 ,栈中剩余1   2   3 ,栈顶元素为3
    System.out.println(s.pop());  // 3出栈 ,栈中剩余1  2  栈顶元素为3 if(s.empty()){
    if (s.empty()) {
        System.out.println("栈空");
    } else {
        System.out.println(s.size());
    }
}

三、栈的模拟实现

在这里插入图片描述
从上图中可以看到, Stack继承了Vector VectorArrayList类似,都是动态的顺序表,不同的是Vector是线程安全的。

1.入栈图解

在这里插入图片描述

2.出栈图解

在这里插入图片描述

3.数组实现的栈

代码如下(示例):

package stackdemo;

import java.util.Arrays;

public class MyStack {
    //用什么来组织呢?
    // 数组、链表
    // 目前先用数组

    //先创建数组
    public int[] elem;
    public int usedSize;//表示有效个数,也可以当下标使用
    public static final int DEFAULT_CAPACITY = 10;

    public MyStack() {
        //初始化数组容量
        this.elem = new int[DEFAULT_CAPACITY];
    }

    //压栈  入栈
    public void push(int val){
        if (isFull()){
            //扩容
            this.elem = Arrays.copyOf(elem, 2 * elem.length);

        }
        elem[usedSize++] = val;
    }

    /**
     * 判断数组是否满了
     */
    public boolean isFull(){
        return usedSize == this.elem.length;
    }


    /**
     * 出栈
     * @return
     */
    public int pop(){
        if (isEmpty()){
            throw new EmptyStackException("栈为空");
        }
        usedSize--;
        return elem[usedSize];
    }

    public boolean isEmpty(){
        return usedSize == 0;
    }

    public int peek(){
        if (isEmpty()){
            throw new EmptyStackException("栈为空");
        }
        return elem[usedSize - 1];
    }
}

测试类

import stackdemo.MyStack;

import java.util.LinkedList;
import java.util.Stack;

public class Test {
    public static void main(String[] args) {
        LinkedList<Integer> stack = new LinkedList<>();
        stack.push(1);
        stack.push(2);
        stack.push(3);

        System.out.println(stack.pop());

        System.out.println(stack.peek());

    }

    public static void main01(String[] args) {
//        Stack<Integer> stack = new Stack<>();
        MyStack stack = new MyStack();
        //向栈里面添加元素

        stack.push(12);
        stack.push(23);
        stack.push(34);
        stack.push(45);
        //先进后出

        //出栈,有两个方法
        // pop 弹出,有一个返回值 直接从栈里面删除元素

        int ret = stack.pop();
        System.out.println(ret);//45

        // peek 也有返回值
        // peek 只是获取栈顶元素 ,不删除
        // 元素还在栈里面
        int peek = stack.peek();
        System.out.println(peek);

        // 判断栈空不空
        System.out.println(stack.isEmpty());

    }
}

4.链表实现的栈

像数组那样在尾部插入删除。大家都知道链表效率低在查询,而查询到尾部效率很低,就算用了尾指针,可以解决尾部插入效率,但是依然无法解决删除效率(删除需要找到前驱节点),还需要双向链表。前面虽然详细介绍过双向链表,但是这样未免太复杂!

所以我们先采用带头节点的单链表在头部插入删除,把头当成栈顶,插入直接在头节点后插入,删除也直接删除头节点后第一个节点即可,这样就可以完美的满足栈的需求。

代码如下(示例):

package stackdemo;


public class lisStack<T> {
    
    static class Node<T> {
        T data;
        Node next;

        public Node() {
        }

        public Node(T value) {
            this.data = value;
        }
    }

    int length;
    Node<T> head;//头节点

    public lisStack() {
        head = new Node<>();
        length = 0;
    }

    boolean isEmpty() {
        return head.next == null;
    }

    int length() {
        return length;
    }

    public void push(T value) {//近栈
        Node<T> team = new Node<T>(value);
        if (length == 0) {
            head.next = team;
        } else {
            team.next = head.next;
            head.next = team;
        }
        length++;
    }

    public T peek() throws Exception {
        if (length == 0) {
            throw new Exception("链表为空");
        } else {//删除
            return (T) head.next.data;
        }
    }

    public T pop() throws Exception {//出栈
        if (length == 0) {
            throw new Exception("链表为空");
        } else {//删除
            T value = (T) head.next.data;
            head.next = head.next.next;//va.next
            length--;
            return value;
        }
    }

    public String toString() {
        if (length == 0) {
            return "";
        } else {
            String va = "";
            Node team = head.next;
            while (team != null) {
                va += team.data + " ";
                team = team.next;
            }
            return va;
        }
    }
}



5.push(链表实现)

push插入

与单链表头插入一致,如果不太了解可以看看前面写的线性表有具体讲解过程。

和数组形成的栈有个区别,链式实现的栈理论上栈没有大小限制(不突破内存系统限制),不需要考虑是否越界,而数组则需要考虑容量问题。

  • 如果一个节点team入栈:
  • 空链表入栈head.next=team;
  • 非空入栈team.next=head.next;head.next=team;

在这里插入图片描述

6.pop(链表实现)

pop弹出

与单链表头删除一致,如果不太了解请先看前面单链表介绍的。

和数组同样需要判断栈是否为空,如果节点team出栈:head指向team后驱节点。

在这里插入图片描述

四、栈的应用场景

1.改变元素的序列

  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

  2. 一个栈的初始状态为空。现将元素1、2、3、4、5、A、 B、C、 D、 E依次入栈 ,然后再依次出栈 ,则元素出栈的顺 序是( )。

    A: 12345ABCDE
    B: EDCBA54321
    C: ABCDE12345
    D: 54321EDCBA

2.将递归转化为循环

2.1 递归方式

思路解析:

  • 如果 head 不为 null,递归调用 printList(head.next) 先递归到链表的末尾。
  • 当递归回溯时,打印当前节点 head 的值。

工作原理:

  • 当 printList(head.next) 运行到链表末尾时,开始逐层回溯。
  • 每次回溯时,会依次打印每个节点的值,实现了链表的逆序输出。
// 递归方式
void printList(Node head) {
    if (null != head) {
        printList(head.next);
        System.out.print(head.val + " ");
    }
}


2.2 循环方式

思路解析:

  • 如果 head 为 null,直接返回。
  • 使用一个栈 s 来存储链表中的节点。
  • 遍历链表,将每个节点依次压入栈中。
  • 最后,依次弹出栈中的节点并打印其值,实现了链表的逆序输出。

工作原理:

  • 遍历链表的过程中,将节点依次压入栈中,因为栈的特性是后进先出(LIFO)。
  • 当遍历完成后,栈中的节点顺序是链表的逆序。
  • 依次弹出栈中的节点并打印,即可实现链表元素值的逆序输出。
// 循环方式
void printList(Node head) {
    if (null == head) {
        return;
    }

    Stack<Node> s = new Stack<>();
// 将链表中的结点保存在栈中
    Node cur = head;
    while (null != cur) {
        s.push(cur);
        cur = cur.next;
    }
    // 将栈中的元素出栈
    while (!s.empty()) {
        System.out.print(s.pop().val + " ");
    }
}

五、了解中缀表达式、后缀表达式

  • 下面以 a + b * c + ( d * e + f ) * g 为例子

  • 讲下应该怎么把中缀表达式转换成后缀表达式。

  • 按先加减后乘除的原则给表达式加括号

  • 结果:((a+(bc))+(((de)+f)*g))

  • 由内到外把每个括号里的表达式换成后缀

  • 最终结果:a b c * + d e * f + g * +

  • 这样就得到了中缀表达式转后缀表达式的最终结果。

  • 此法应付考试有神效。

总结

LinkedKist 就可以当做栈来使用

  • 递归方式:简单、优雅,但可能会面临栈溢出的风险,特别是在链表非常长的情况下。
  • 循环方式:使用了额外的栈来辅助逆序输出,空间复杂度略高,但是可以避免递归深度过深导致的栈溢出问题。

在这里插入图片描述

在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值