【数据结构篇】线性表2 —— 栈和队列

news2025/1/26 14:42:23

前言:上一篇我们介绍了顺序表和链表

(https://blog.csdn.net/iiiiiihuang/article/details/132615465?spm=1001.2014.3001.5501),

这一篇我们将介绍栈和队列,栈和队列都是基于顺序表和链表来实现的

目录

栈(Stack)

什么是栈 ?

栈的方法 和 使用

栈的模拟实现 

先初始化一下栈 

往栈里插入元素 (push)

栈是否为空(empty)

弹出栈顶元素(删除)(pop)

获取栈顶元素 (peek)

模拟实现完整代码 

栈的应用场景

 1. 改变元素的序列

2. 将递归转化为循环

补充 :

队列(Queue) 

什么是队列 ?

队列的方法 

队列模拟实现 

初始化 

offer 

poll

peek

完整代码(链式队列) 

循环队列 

认识循环队列 

如何把数组看作是一个循环呢?(rear 从 7 --> 0) 

设计一个循环队列 (https://leetcode.cn/problems/design-circular-queue/)

初始化 

判断队列是否是空的,和是否是满的 

入队 

 出队

得到队头元素

得到队尾元素

完整代码: (循环队列)

双端队列 (Deque)

(面试题) 1. 用队列实现栈   2. 用栈实现队列

 1. 用队列实现栈

代码实现:

入栈

出栈 

获取栈顶元素 

 判断栈空

完整代码 

 2. 用栈实现队列 

我直接上代码了: 

 


 

栈(Stack)

什么是栈 ?

栈:一种特殊的线性表,其只允许在固定的一端进行插入和删除元素操作。进行数据插入和删除操作的一端称为栈顶,另一端称为栈底。栈中的数据元素遵守后进先出LIFO(Last In First Out)的原则。

后进先出原则:先进的后出,后进的先出  。我们举一个生活中的例子

比如这有个圆筒,一端封闭,一端开口,然后往圆筒里放乒乓球:

    

你想拿乒乓球,先拿到的肯定是后放的,那如果你想要拿到最底下的,你肯定得把上面的都拿走才行。也就是 栈的后进先出原则 (先进的后出,后进的先出)

栈的方法 和 使用

 栈的方法很少,只有这些: 

使用 (没什么好说的😁😁)

栈的模拟实现 

 栈这种数据结构可以用数组来实现,也可以用链表来实现,这里我们用数组实现。

先初始化一下栈 

往栈里插入元素 (push)

和顺序表异曲同工,先看看满没满,满了要扩容,再入栈 

栈是否为空(empty)

弹出栈顶元素(删除)(pop)

就很简单 (顺序表那搞懂,这些方法很简单的)

 

获取栈顶元素 (peek)

 

模拟实现完整代码 

import java.util.ArrayList;
import java.util.Arrays;

/**
 * @Author: iiiiiihuang
 */
public class MyStack {
    private int[] elem;
    private int usedSize;
    private static final int DEFAULT_CAPACITY = 10;//默认容积
    public MyStack() {
        this.elem = new int[DEFAULT_CAPACITY];
    }

    /**
     * 往栈里插入元素
     * @param val
     */
    public void push(int val) {
        isFull();
        this.elem[usedSize] = val;
        usedSize++;
    }
    private void isFull() {
        if(usedSize == this.elem.length) {
            this.elem = Arrays.copyOf(this.elem, 2*this.elem.length);
        }
    }

    /**
     * 弹出栈顶元素
     * @return
     */
    public int pop() {
        if(empty()) {
            throw new EmptyException("栈为空");
        }
        int pop = this.elem[usedSize - 1];
        usedSize--;
        return pop;
    }

    /**
     * 获取栈顶元素
     * @return
     */
    public int peek() {
        if(empty()) {
            System.out.println("栈为空");
            return 0;
        }
        int peek = this.elem[usedSize - 1];
        return peek;
    }

    /**
     * 栈是否为空
     * @return
     */
    public boolean empty() {
        return usedSize == 0;
    }
}

栈的应用场景

 1. 改变元素的序列

我们去牛客上找两道题做做 

根据我们上面介绍,栈 遵循 先进后出原则, 我们可以得出答案,不可能的出栈顺序是 D 选项

因为F,E出来了,下一个出来的必须是 D,这种情况下C不可能比D先出。 

2. 将递归转化为循环
 

比如逆序打印链表: 

  1.递归方法 

 

 2.循环方法 

 

运行结果 

 

 

补充 :

用数组实现栈,出栈,入栈,都是O(1),

那用链表如何实现栈呢?

1.假设用单链表(单向)实现栈: 

  尾插: 

      入栈:尾插法 入栈是 O(n)

      出栈:因为栈后进的先出,这就相当于删除链表的尾巴节点,时间复杂度还是O(n);

 头插: 

      入栈:头插法 O(1)

      出栈:相等于删除头节点,时间复杂度O(1)

2.假设用双向链表实现栈:  (前后都能找到)

尾插: 

      入栈:尾插法 入栈是 O(1)

      出栈:时间复杂度还是O(1),

 头插: 

      入栈:头插法 O(1)

      出栈:时间复杂度O(1)

上述我们可知,用链表实现栈时,双向链表实现栈是最好的,不管从那边入栈,出栈 时间复杂度都是O(1)。

那使用链式栈时就可以这样使用:如果这是个栈,那peek的内容就是 4. 

 

是 4. 

 

那为啥会这样,你看LinkedList,里本来就有栈里的一些方法,push..... 

 

push 用的是头插法 

 

 

所以如果是顺序栈,你可以用Stack, 如果是链式栈,你就用LinkedList 

 

 

队列(Queue) 

什么是队列 ?

 队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出 FIFO(First In First Out)

入队列:进行插入操作的一端称为队尾(Tail/Rear) 出队列:进行删除操作的一端称为队头

举例说明: 

假如有一条单行隧道,车辆必须从入口进 (入队列),从出口出(出队列)。那先驶出隧道的车辆肯定是先进去的,后驶出的肯定是后进去的,这就是 队列的先进先出原则

通过上述介绍,我们发现我们可以把 双向链表当成是一个 队列 ,(单链表也行,双链表更简单)

队尾进,队头出,时间复杂度O(1)

 

 ps:Queue 普通队列,Deque 双端队列(两边都可以进出),LinkedList 可以当作普通队列,双端队列,链表,栈(究极打工人😭😭😭)

队列的方法 

在Java中,Queue是个接口,底层是通过链表实现的。 

队列方法少的嘞 

 

offer (入队列) 

 

peek 

 

poll 

 

......... 

add,remove ,element (和peek一个意思)是一组 

offer,  poll,  peek 是一组

 

 

队列模拟实现 

队列的实现使用 顺序结构 和 链式结构 都行,但链式结构更简单一点。

我们这里拿双链表去写 (前面的链表学会了,这里很简单的,信手拈来)

初始化 

 

 

offer 

 

调试一看 插进去了 

 

poll

 

 

 

 

peek

 

 

 

完整代码(链式队列) 

 

/**
 * @Author: iiiiiihuang
 */
public class MyQueue {
    static class ListNode {
        private int val;
        private ListNode prev;
        private ListNode next;
        public ListNode(int val) {
            this.val = val;
        }
    }
    private ListNode front;//头
    private ListNode rear;//尾
    private int usedSize;//计数

    /**
     * 插入--- 头插法
     * @param val
     */
    public void offer(int val) {
        ListNode node = new ListNode(val);
        if(front == null) {
            front = node;
            rear = node;
        } else {
            node.next = front;
            front.prev = node;
            node = front;
        }
        usedSize++;
    }

    /**
     * 删除 -- 尾删
     * @return
     */
    public int poll() {
        if(front == null) {
            throw new EmptyException("队列为空");
        }
        if(front == rear) {
            int poll = front.val;
            front = null;
            rear = null;
            usedSize--;
            return poll;
        }
        int poll = rear.val;
        rear = rear.prev;
        rear.next = null;
        usedSize--;
        return poll;
    }

    /**
     * 获取队头
     * @return
     */
    public int peek() {
        if(front == null) {
            throw new EmptyException("队列为空");
        }
        return rear.val;
    }
}

 

循环队列
 

实际中我们有时还会使用一种队列叫循环队列
环形队列通常使用数组实现。 

认识循环队列 

 

 

。。。。。 

 

这就有意思了 ,

front 和 rear 第一次在一起的时候 队列 是空的,第二次 在一起的时候是队列是满的

那这怎么判断呢,其实有很多方法去判断

 1.计数  usedSize == len

 2.标记,flag

 3.浪费一个空间来区分(常用) 

 浪费一个空间,就类似下面这样

 

 

如何把数组看作是一个循环呢?(rear 从 7 --> 0) 

就是把 rear ++  换成 (rear + 1 )% len  (余数肯定都是小于 len的(len 数组长度)),rear++ 会越界的 (看不懂自己带数据算算)

 

设计一个循环队列 (https://leetcode.cn/problems/design-circular-queue/)

根据这个来 

 

 

初始化 

 

判断队列是否是空的,和是否是满的 

 

 

入队 

 

 

 出队

 

 

得到队头元素

 

得到队尾元素

 

不能直接rear - 1哦

 那现在我们放到力扣里运行一下(链接在上边)

 错了 !!! 为什么,我们看看我什么

 

数组容量是 3 ,预期结果成功 插入了 3个,而我们的程序,只能插入两个,因为我们的浪费掉了一个空间 。

 

所以我们应该多给一个空间: 

 

在运行一下 ,通过了,当然你要是觉得这样奇怪,你也可以用 usedSize,很多方法啦😎😎

 

完整代码: (循环队列)

class MyCircularQueue {

    private int[] elem;
    private int front;//头
    private int rear;//尾
    public MyCircularQueue(int k) {
        this.elem = new int[k + 1];
    }
    
    public boolean enQueue(int value) {
        if(isFull()) {
            return false;
        }
        elem[rear] = value;
        rear = (rear + 1) % elem.length;
        return true;
    }
    
    public boolean deQueue() {
        if(isEmpty()) {
            return false;
        }
        front = (front + 1) % elem.length;
        return true;
    }
    
    public int Front() {
        if(isEmpty()) {
            return -1;
        }
        int ret = elem[front];
        return ret;
    }
    
    public int Rear() {
         if(isEmpty()) {
            return -1;
        }
        int index = (rear == 0) ? elem.length - 1 : rear - 1;//不能直接rear - 1哦
        return elem[index];
    }
    
    public boolean isEmpty() {
        return front == rear;
    }
    
    public boolean isFull() {
        if((rear + 1) % this.elem.length == front) {
            return true;
        }
        return false;
    }
}

 

双端队列 (Deque)
 

 双端队列(deque)是指允许两端都可以进行入队和出队操作的队列,deque 是 “double   ended queue” 的简称。
那就说明元素可以从队头出队和入队,也可以从队尾出队和入队

Deque是一个接口,使用时必须创建LinkedList的对象 。

 在实际工程中,使用Deque接口是比较多的,栈和队列均可以使用该接口

1. Deque<Integer> stack = new ArrayDeque<>();//双端队列的线性实现(数组)
2. Deque<Integer> queue = new LinkedList<>();//双端队列的链式实现 

 

(面试题) 1. 用队列实现栈   2. 用栈实现队列

 1. 用队列实现栈

225. 用队列实现栈 - 力扣(LeetCode) 

1.首先一个栈是不够用的,它俩的出栈原则不一样,栈先进后出,队列先进先出,一个队列实现不了呀 ,对不上。

2.所以我们需要两个队列,当 2 个队列都为空的时候,说明模拟栈为空。

 

3.当要出栈的时候,(出 45 ),我们要把 12,23,34,入到第二个队列里去,然后再把第一个队列里的 45 出队列 ,

结论就是,”出栈“的时候出不为空的 队列,出队列size - 1 个(俩队列轮流),最后那一个就是我们要 ”出栈“的元素

 

 4.“入栈” 的时候入到不为空的队列

 思路有了,我们开始写代码:

代码实现:

入栈

 

出栈 

 用for 循环的时候注意一下,不要写成 i < queue.size() - 1,因为queue.size()一直在变。你先记录一下

 

获取栈顶元素 

 我们可以定义一个临时量,来存放栈顶元素

 

里面的元素一直被覆盖,那最后出队列的元素,就覆盖了之前的全部元素,就是栈顶元素。 

 

 注意:这里可别写成 queue1.offer(queue2.poll), 跳两回啦。

 判断栈空

完美通过 

 

完整代码 

 

import java.util.LinkedList;
import java.util.Queue;

/**
 * @Author: iiiiiihuang
 */
public class MyStack {
    private Queue<Integer> queue1;
    private Queue<Integer> queue2;
    public MyStack() {
        queue1 = new LinkedList<>();
        queue2 = new LinkedList<>();
    }

    /**
     * 入栈操作
     * @param x
     */
    public void push(int x) {
        if(!queue1.isEmpty()){
            queue1.offer(x);
        } else if(!queue2.isEmpty()) {
            queue2.offer(x);
        }else {
            queue1.offer(x);
        }
    }

    /**
     * 出栈
     * @return
     */
    public int pop() {
        //先判断栈空不空
        if(empty()) {
            return -1;
        }
        if(!queue1.isEmpty()) {
            int len = queue1.size() - 1;
            while (len > 0) {
                int tmp = queue1.poll();
                queue2.offer(tmp);
                len--;
            }
            return queue1.poll();
        } else {
            int len = queue2.size() - 1;
            while (len > 0) {
                int tmp = queue2.poll();
                queue1.offer(tmp);
                len--;
            }
            return queue2.poll();
        }
    }

    /**
     * 获取栈顶元素
     * @return
     */
    public int top() {
        int tmp = -1;
        if(empty()) {
            return -1;
        }
        if(!queue1.isEmpty()) {
            int len = queue1.size();
            while (len > 0) {
                tmp = queue1.poll();
                queue2.offer(tmp);
                len--;
            }
            return tmp;
        } else {
            int len = queue2.size();
            while (len > 0) {
                tmp = queue2.poll();
                queue1.offer(tmp);
                len--;
            }
            return tmp;
        }
    }

    public boolean empty() {
        return queue1.isEmpty() && queue2.isEmpty();
    }
}

 

 2. 用栈实现队列 

232. 用栈实现队列 - 力扣(LeetCode)

思路差不多 

1.两个栈实现,同理 

2.入队的时候 把所有元素 全部放到第一个栈当中

3.出队的时候 把所有的元素 全部倒到第二个栈当中,然后出第二个栈顶元素就行了.(记得判空)

我直接上代码了: 

import java.util.Stack;

/**
 * @Author: iiiiiihuang
 */
public class MyQueue {
    private Stack<Integer> stack1;
    private Stack<Integer> stack2;

    public MyQueue() {
        stack1 = new Stack<>();
        stack2 = new Stack<>();
    }

    public void push(int x) {
        stack1.push(x);
    }

    public int pop() {
        if(empty()) {
            return -1;
        }
        if(stack2.isEmpty()) {
            while (!stack1.isEmpty()) {
                stack2.push(stack1.pop());
            }
        }
        return stack2.pop();
    }

    public int peek() {
        if(empty()) {
            return -1;
        }
        if(stack2.isEmpty()) {
            while (!stack1.isEmpty()) {
                stack2.push(stack1.pop());
            }
        }
        return stack2.peek();
    }

    public boolean empty() {
        return stack1.isEmpty() && stack2.isEmpty();
    }
}

 

 

先别急着走,点个关注先,👀👀👀

点赞,评论,收藏,支持一下ヾ(≧▽≦*)oヾ(≧▽≦*)o

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/979530.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Metinfo4.0逻辑漏洞

搭建网站 MetInfo历史版本与文件&#xff0c;这里下载 需要进行安装 漏洞复现 点击会员中心进行注册 点击找回密码&#xff0c;输入刚刚创建的账号&#xff0c;然后抓包 登录后修改基本信息 直接抓包修改admin用户的密码 使用admin用户和密码654321&#xff0c;发现登录成功…

波奇学C++:继承

继承是为了复用代码&#xff0c;成员的变量或者成员函数 class Person { public:protected:string _name"li";int _age1; }; class Student :public Person { public:void print(){cout << _age;} protected:int _stuid2; }; 子类student公有继承基类Person,…

可视化工具Datart踩(避)坑指南(1)——不可复用的图表

作为目前国内开源版本最好用的可视化工具&#xff0c;Datart无疑是低成本高效率可供二开的可视化神兵利器。当然&#xff0c;免费的必然要付出一些踩坑的代价。本篇我们来讲一讲可视化工具Datart踩&#xff08;避&#xff09;坑指南&#xff08;1&#xff09;——不可复用的图表…

【小吉测评】高效简洁的数据库管控平台—CloudQuery

文章目录 &#x1f384;CloudQuery是什么&#x1f6f8;CloudQuery支持的数据源类型&#x1f354;CloudQuery社区地址&#x1f33a;如何使用&#x1f6f8;参考官方文档&#x1f6f8;参考视频教程&#x1f388;点击免费下载&#x1f388;立即下载即可&#x1f388;使用服务器完成…

基于Mendix移动原生的离线应用

一、前言 不同行业的企业会有特殊的业务场景&#xff0c;比如某些制造业的企业的工厂是物理隔离的&#xff0c;但工程师需要拿着平板输入很多生产数据&#xff1b;某些煤炭和矿业企业&#xff0c;在实际的工作区都是比较偏远&#xff0c;信号比较差&#xff0c;但是又需要用手…

gt基础教程

每日练习 吉他-结构 吉他-品 吉他谱 六线谱 调音器 试炼导航 每日练习 流程1&#xff1a; 右手拨弦: 大拇指单独练习 控制654弦&#xff0c;每根弦拨4下 6弦4下 5弦4下 4弦4下食指无名指练习 控制321弦&#xff0c;321、321练习&#xff0c; 然后321、123练习&#xff0…

指令系统(408)

一、拓展操作码指令格式 【2017 统考】某计算机按字节编址&#xff0c;指令字长固定且只有两种指令格式&#xff0c;其中三地址指令29条、二地址指令107条&#xff0c;每个地址字段6位&#xff0c;则指令字长至少应该是&#xff08; A&#xff09; A、24位 B、26位 …

离散型行业与MES系统——密不可分的关系

离散型行业通常指的是制造业中的一类&#xff0c;其中产品制造过程是通过离散的步骤和阶段完成的&#xff0c;而不是连续不断的过程。这些离散型行业包括汽车制造、电子制造、航空航天、医药制造、机械制造等。在这些行业中&#xff0c;产品通常是由不同的零部件和组件组装而成…

DM8 安装手册(官方原版包含卸载)

DM8 安装手册 一 .安装简介1.1 DM 产品的构成DM Standard Edition 标准版DM Enterprise Edition 企业版DM Security Edition 安全版 DM 产品主要由数据库服务器和客户端程序两大部分组成。其中数据库服务器包括多种操作系统下的版本&#xff0c;主要有&#xff1a;DM 客户端程序…

TikTok美国市场现状如何?

众所周知&#xff0c;我国是世界上人口最多的国家&#xff0c;近年来经济水平也迅速提高&#xff0c;因此消费水平自然较高。但是实际上消费水平最高的国家是美国&#xff0c;人口仅超过3亿。目前在这个跨境电商人人都想尝试的时代&#xff0c;美国这个国家更是成为了跨境电商卖…

typeof 在TypeScript中和JavaScript中的区别

前言 在TypeScript中和JavaScript中都有typeOf&#xff0c;但是作用用法却大有不同。 js的typeof 一、typeof用来判断数据类型返回结果&#xff1a; 基本数据类型&#xff1a;string&#xff0c;number&#xff0c;boolean,undefined 引用数据类型&#xff1a;object …

yocto stm32mp1集成ros

yocto stm32mp1集成ros yocto集成ros下载meta-rosyocto集成rosrootfs验证 yocto集成ros 本章节介绍yocto如何集成ros系统用来作机器人开发。 下载meta-ros 第一步首先需要下载meta-ros layer&#xff0c;meta-ros的链接如下&#xff1a;https://github.com/ros/meta-ros/tre…

环信uni-app-demo 升级改造计划——单人多人音视频通话(三)

前序文章&#xff1a; 环信 uni-app Demo升级改造计划——Vue2迁移到Vue3&#xff08;一&#xff09; 环信即时通讯SDK集成——环信 uni-app-demo 升级改造计划——整体代码重构优化&#xff08;二&#xff09; 概述 在将声网 uni-app 音视频插件正式集成进入环信的 uni-app…

毫米波水位监测仪:实时监测水体水位变化

水位监测是一项关键的技术&#xff0c;用于实时监测水体的水位变化&#xff0c;对于水利工程、自然灾害预防和水资源管理都具有重要的意义。通过在各关键节点安装毫米波水位监测仪&#xff0c;可对水位情况进行实时监测&#xff1b;当水位超过阈值时&#xff0c;智能监测仪器将…

一份超预期的期中成绩,拨开百果园“高价值迷雾”

文 | 螳螂观察 作者 | 青月 步入2023年&#xff0c;经济复苏、消费增长趋势显现&#xff0c;但实体店还未完全突破桎梏。 不过&#xff0c;即使是在这样的市场环境中&#xff0c;年初成功“上岸”&#xff0c;估值一度达百亿的百果园&#xff0c;依旧交出了一份营收净利双增…

二蛋赠书一期:《快捷学习Spring》

文章目录 前言活动规则参与方式本期赠书《快捷学习Spring》关于本书作者介绍内容简介读者对象 结语 前言 大家好&#xff01;我是二蛋&#xff0c;一个热爱技术、乐于分享的工程师。在过去的几年里&#xff0c;我一直通过各种渠道与大家分享技术知识和经验。我深知&#xff0c…

【C++】函数重载 ④ ( 函数指针定义的三种方式 | 直接定义函数指针 | 通过 函数类型 定义 函数指针 | 通过 函数指针类型 定义 函数指针 )

文章目录 一、函数指针定义方法1、直接定义函数指针2、通过 函数类型 定义 函数指针3、通过 函数指针类型 定义 函数指针4、代码示例 - 不同方式定义函数指针 博客总结 : 重载函数 : 使用 相同 的 函数名 , 定义 不同 的 函数参数列表 ;判定标准 : 只有 函数参数 的 个数 / 类…

给抖音达人推商品需要注意什么?抖店商家注意了,教你几个技巧

我是王路飞。 找达人带货这种玩法&#xff0c;虽然商家要给带货达人佣金&#xff0c;相当于你的利润变少了。 但是你要明白一件事&#xff0c;我们做抖店也不是做着玩的&#xff0c;而是奔着长线去玩的&#xff0c;所以长久稳定才是我们需要的。 而相比较自然流量&#xff0…

SecureCRT安装、汉化、上传、美化

文章目录 SecureCRT安装、汉化、美化一、SecureCRT介绍二、SecureCRT下载三、SecureCRT汉化四、SecureCRT连接五、SecureCRT上传第一种方法&#xff1a;用SFTP 传输文件第二种方法&#xff1a;WinSCP传输文件 六、SecureCRT美化 SecureCRT安装、汉化、美化 一、SecureCRT介绍 …

Source Insight 宏-添加单行的c注释

今天写代码的时候突然想到在代码的上一空行添加一对 /* */ 来添加注释&#xff0c;或者单独注释一行代码&#xff0c;而且是用 c 的注释方式&#xff0c;即使用 /**/&#xff0c;如想要在光标处添加 /**/ 或者注释掉光标所在的行&#xff0c;如&#xff1a; 实际的效果就是这样…