当前位置: 首页 > news >正文

《数据结构》队列及其经典面试题

前言

上一篇讲了栈和栈的经典面试题,链接如下:

栈与栈的经典面试题

其实栈和队列是一码事,都是对只能再线性表的一端进行插入和删除。

因此,其实栈和队列可以互相转换!

一、队列的特点

先进先出的数据结构,元素从“队尾”添加到队列中,元素从“队首”出队列 (FIFO)

在这里插入图片描述


二、队列的实现

1.基于链表实现队列

现实生活中,有各式各样的“排队”操作。

同样的,队列也有基于数组实现的队列和基于链表实现的队列。

由于出队操作只能在队列的头部进行,若采用数组的方案,每次出队一个元素就得搬移剩下的所有元素向前移动一个单位。

此时采用链表的方案更加适合队列的结构。

2.核心操作

  • E poll() : 出队

  • offer(E e) : 入队


三、双端队列

双端队列(deque)是指允许两端都可以进行入队和出队操作的队列,deque 是 “double ended queue” 的简称。

那就说明元素可以从队头出队和入队,也可以从队尾出队和入队。

Deque -> Queue的子接口:

在这里插入图片描述

小技巧:

  • 大家以后无论使用的是栈还是队列,统一使用双端队列接口!
  • 不推荐使用Stack这个类,这个类已经是时代的弃子,效率很低,都是同步操作。
  • 双端队列的一个常用子类就是LinkedList。

在这里插入图片描述
在这里插入图片描述


四、循环队列

1.应用场景

  • 操作系统的生产消费者模型
  • MySQL数据库的InnoDB存储引擎中的redo日志。

2.实现

  1. 基本都是使用长度固定的数组实现,数组在实现队列时,若从数组头部删除元素需要频繁的移动后面的元素,带来的效率比较低。

  2. 出队和入队操作,使用两个引用,一个head,一个tail,添加元素在数组尾部添加,删除元素只需要移动head引用指向的地址即可(逻辑删除)。

循环队列就是使用长度固定的数组来实现,数组头部就是队首(head),数组的尾部就时队尾(tail)。数组[head…tail]是循环队列的有效元素。

例如:

在这里插入图片描述

  • head永远指向循环队列的第一个元素
  • tail永远指向循环队列有效元素的后一个位置

循环队列在删除元素时,不需要进行数据的搬移,当有新的元素在添加时就会覆盖掉之前的元素。

所谓的循环队列指的是当head或者tail引用走到数组末尾时,下一次再继续向后移动,其实是返回数组的头部继续操作。

3.判空和判满

由于tail指向的是有效数组后一个位置,所以当tail走到如图所示的head == tail 情况时:

在这里插入图片描述

此时我们没法区分当前循环队列到底是空还是满!!!

所以在循环队列中,需要浪费一个空间来判断队列是否已满,如下图所示:

在这里插入图片描述

结论:

  1. 此时当 (tail+1)%n == head 时,认为此时循环队列已满。
  2. head和tail的移动不能简单的+1,而要使用取模操作,取数组长度。
  3. 当head == tail 时 ,认为此时循环队列为空。

4.最后一个元素的索引

  • 除了tail这个引用指向0这个位置以外,其他情况的最后一个索引 = tail - 1
  • 当 tail = 0 时,最后一个元素就在数组的末尾,索引 = data.length - 1

在这里插入图片描述

代码如下:

int lastIndex = tail == 0 ? data.length -1 : tail - 1

五、队列的常见问题

1.用队列实现栈

链接如下:225.用队列实现栈

在这里插入图片描述


解题思路:

思路1:(双队列思路)

这个问题的本质和双栈实现最小栈是相同的思路,一定要保证其中一个队列就是进行实际元素存储的,另一个队列就是作为辅助操作。

q1永远是存储元素的队列,新元素添加到q2中,将此时q1中的所有元素出队再入队q2恰好就能实现添加顺序和出队顺序相反的操作

  1. 新元素永远入q2
  2. 将老元素q1依次出队再入q2 (这样就交换了元素的先后顺序)
  3. q1和q2交换引用

在这里插入图片描述

  • 其中一个队列q1永远都是存储元素的队列,栈的pop就是s1的poll,栈的peek就是s1的peek,栈的push就是s1的offer。 (所以整个题的核心操作就是push()

  • 以保证s1和栈的操作一一对应。

  • 另外一个队列就是作为辅助操作。

代码如下:

class MyStack {
    Deque<Integer> temp;
    Deque<Integer> q1;
    Deque<Integer> q2;
    public MyStack() {
        q1 = new LinkedList<>();
        q2 = new LinkedList<>();
    }
    
    public void push(int x) {
        q2.offer(x);
        while(!q1.isEmpty()){
            q2.offer(q1.poll());
        }
        temp = q1;
        q1 = q2 ;
        q2 = temp;
    }
    
    public int pop() {
        return q1.poll();
    }
    
    public int top() {
        return q1.peek();
    }
    
    public boolean empty() {
        return q1.isEmpty();
    }
}

思路2:(单队列思路)

在这里插入图片描述

  1. 先记录下当前队列中的元素个数n
  2. 将新元素直接入队
  3. 为了让新元素变成队首元素,连续出队n次(新元素的之前的所有元素都出队列,此时新元素变成了队首元素),再依次入队n次即可。

代码如下:

class MyStack {
    private Deque<Integer> queque;
    private int length=0;//1.先记录一下栈此时的元素个数
    public MyStack() {
        queque=new LinkedList<>();
    }

    public void push(int x) {
        queque.offer(x);//2.新元素直接入队
        //3.把新元素之前的元素挨个出队再入队,此时最新的元素就是队头元素
        for (int i=0;i<length;i++){
            queque.offer(queque.poll());
        }
        length++;

    }

    public int pop() {
        length--;
        return queque.poll();

    }

    public int top() {
        return queque.peek();
    }

    public boolean empty() {
        return queque.isEmpty();
    }
}

2.用栈实现队列

链接如下:用栈实现队列

在这里插入图片描述

解题思路:

思路1:(入队 - 时间复杂度为O(n),出队 - O(1))

定义s1永远是保存元素的栈

  1. 先将s1中的现有元素依次弹出放入s2
  2. 将新元素入s1 => 此时这个新元素就变成了s1的栈底(队尾元素)
  3. 将s2中的元素再依次弹回来(先进先出)

在这里插入图片描述

代码如下:

class MyQueue {
    Deque<Integer> s1;
    Deque<Integer> s2;
    public MyQueue() {
        s1 = new LinkedList<>();
        s2 = new LinkedList<>();
    }
    
    public void push(int x) {
        while(!s1.isEmpty()){
            s2.push(s1.pop());
        }
        s1.push(x);
        while(!s2.isEmpty()){
            s1.push(s2.pop());
        }
    }
    
    public int pop() {
        return s1.pop();
    }
    
    public int peek() {
        return s1.peek();
    }
    
    public boolean empty() {
        return s1.isEmpty();
    }
}

思路2:(入队 - 时间复杂度为O(1),出队-摊还复杂度O(1))

  1. push是正常push的,核心操作在pop里面
  2. push进s1的元素依次出栈再入s2栈的时候,出入顺序就会颠倒
  3. 根据上述特性,把s2的栈顶当作队首就行了,因为队列都是队首出队的。
  4. 每次pop先判断当前s2是否为空,若为空,则把s1中的元素全部出栈并且压进s2里面,此时s2的栈顶就是队首元素(先进先出)。若不为空,证明上一轮push进来的元素还没pop完,继续pop当前s2的栈顶就行。
class MyQueue {
    Deque<Integer> s1 ;
    Deque<Integer> s2 ;
    public MyQueue() {
        s1 = new LinkedList<>();
        s2 = new LinkedList<>();
    }
    
    public void push(int x) {
        s1.push(x);
    }
    
    public int pop() {
        if(s2.isEmpty()){
            while(!s1.isEmpty()){
                s2.push(s1.pop());
            }
        }
        return s2.pop();
    }
    
    public int peek() {
        if(s2.isEmpty()){
            while(!s1.isEmpty()){
                s2.push(s1.pop());
            }
        }
        return s2.peek();
    }
    
    public boolean empty() {
        return s1.isEmpty() && s2.isEmpty();
    }
}

总结

以上就是队列及其经典面试题的全部内容了,有帮助的话麻烦各位给个三连~~感谢!!!

相关文章:

  • 计算机图形学(十一):真实感图形(光照模型、材质模型)
  • 【云原生】Hadoop HA on k8s 环境部署
  • 四元数是什么
  • 大衣哥家里再添喜事,生产厂家免费送给他一辆新车
  • 爬取疫情数据并存到mysql数据库
  • 场景应用:网络的子网掩码为255.255.240.0,它能够处理的主机数是多少?
  • Qt5开发从入门到精通——第七篇六节( 图形视图—— 图元的旋转、缩放、切变、和位移)
  • 内网穿透工具natapp的注册、下载、安装与使用(详细教程)
  • CDH openssl 安装报错 TXT_DB error number 2
  • 【Linux线程同步专题】一、什么是线程同步、互斥量与死锁
  • 内网渗透-Linux权限维持
  • Git 便捷操作
  • 美国项目管理协会和埃森哲最新报告:越来越多的公司设立首席转型官一职
  • y145.第八章 Servless和Knative从入门到精通 -- 消息系统基础和Eventing及实践(九)
  • CDQ整体二分-三维偏序(陌上花开)
  • iOS 颜色设置看我就够了
  • JavaScript学习总结——原型
  • js继承的实现方法
  • JS数组方法汇总
  • Making An Indicator With Pure CSS
  • node-sass 安装卡在 node scripts/install.js 解决办法
  • passportjs 源码分析
  • socket.io+express实现聊天室的思考(三)
  • vue的全局变量和全局拦截请求器
  • 从 Android Sample ApiDemos 中学习 android.animation API 的用法
  • 从0实现一个tiny react(三)生命周期
  • 观察者模式实现非直接耦合
  • 七牛云假注销小指南
  • 设计模式 开闭原则
  • 数据结构java版之冒泡排序及优化
  • 网页视频流m3u8/ts视频下载
  • 小而合理的前端理论:rscss和rsjs
  • 一起来学SpringBoot | 第三篇:SpringBoot日志配置
  • 一文看透浏览器架构
  • 由插件封装引出的一丢丢思考
  • 策略 : 一文教你成为人工智能(AI)领域专家
  • 好程序员大数据教程Hadoop全分布安装(非HA)
  • ​LeetCode解法汇总2583. 二叉树中的第 K 大层和
  • ​queue --- 一个同步的队列类​
  • ​云纳万物 · 数皆有言|2021 七牛云战略发布会启幕,邀您赴约
  • # 深度解析 Socket 与 WebSocket:原理、区别与应用
  • #pragma once
  • %3cscript放入php,跟bWAPP学WEB安全(PHP代码)--XSS跨站脚本攻击
  • (09)Hive——CTE 公共表达式
  • (10)STL算法之搜索(二) 二分查找
  • (145)光线追踪距离场柔和阴影
  • (Matlab)使用竞争神经网络实现数据聚类
  • (二十五)admin-boot项目之集成消息队列Rabbitmq
  • (教学思路 C#之类三)方法参数类型(ref、out、parmas)
  • (十三)Flask之特殊装饰器详解
  • (原创)Stanford Machine Learning (by Andrew NG) --- (week 9) Anomaly DetectionRecommender Systems...
  • (转)Spring4.2.5+Hibernate4.3.11+Struts1.3.8集成方案一
  • (转)创业的注意事项
  • .NET Core IdentityServer4实战-开篇介绍与规划
  • .NET 跨平台图形库 SkiaSharp 基础应用