数据结构:图文详解 队列 | 循环队列 的各种操作(出队,入队,获取队列元素,判断队列状态)


目录

队列的概念

队列的数据结构

队列的实现

入队

出队

获取队头元素

获取队列长度

循环队列的概念

循环队列的数据结构

循环队列的实现

判断队列是否为空

判断队列是否已满

入队

出队

得到队头元素

得到队尾元素


队列的概念

队列(Queue)是一种数据结构,是一种先进先出(First-In-First-Out,FIFO)的线性数据结构。它只允许在列表的一端进行插入操作(入队),在另一端进行删除操作(出队),即队头进行删除操作,队尾进行插入操作。队列常用的操作有入队(Enqueue)、出队(Dequeue)、获取队头元素(Front/Peek)、获取队列长度(Size/Length)等。

图示如下:

队列的特点是按照元素加入的先后顺序进行操作,先加入队列的元素会先被取出,后加入的元素会后被取出。这种特性常常被用于模拟实际生活中的排队场景,例如银行柜台排队、CPU任务调度等。


队列的数据结构

队列可以用数组或链表来实现。数组实现的队列需要预先分配一定的空间,但是在操作中效率较高;链表实现的队列没有固定的空间限制,但是在操作中可能需要更多的时间和空间。

这里笔者以链表进行演示:

public class MyQueue {
    static class ListNode {
        int val;//存放数据
        ListNode pre;//前驱指针
        ListNode next;//后驱指针
        public ListNode(int val) {
            this.val = val;
        }
    }
    private ListNode head;//记录头节点
    private ListNode last;//记录尾部节点
}

队列的实现

入队

因为我们使用的是链表来模拟实现,所以对于入队的操作就相当于使用尾插法(头插法),入队图示:

当队列为空的时候需要进行特殊处理,不然会造成空指针异常,队列为空的时候让我们的头指针和尾指针都指向这个节点就可以,如果队列不为空就正常的使用尾插法,尾插法图示:

    //入队
    public void enqueue(int val) {
        ListNode newNode = new ListNode(val);
        if (head == null) {
            head = newNode;
            last = newNode;
        }else {
            newNode.pre = last;
            last.next = newNode;
            last = newNode;
        }
    }

出队

队列是个单向的数据结构,对于入队我们使用了尾插法,那么出队就需要使用删除头节点的方法,出队的图示如下:

出队实际上是对于队头节点的删除,所以当队内没有节点的时候我们就不能进行这样的操作,我们直接抛出一个异常,在可以抛出的情况下,我们定义一个 val变量 来存放我们要删除的值,以便返回。当队列只有一个元素的时候要进行特殊处理,不然后面的程序会报空指针异常,这种情况下,我们需要先拿到头节点的值,然后将头节点置空,最后返回这个值;其余情况我们就正常进行删除操作,先拿到头节点的值然后让头节点后移,再让改变现在头节点的前驱指针,让我们要删除的数据无法被访问,最后返回这个值

    //出队
    public int dequeue() {
        if (head == null) {
            throw new ExceptionOfEmpty("队列为空,无法操作");
        }
        int val = -1;
        //当队列只有一个元素的时候要进行特殊处理,不然后面的程序会报空指针异常
        if (head.next == null) {
            val = head.val;
            head = null;
            last = null;
            return val;
        }
        //对于其他正常位置元素的处理
        val = head.val;
        head = head.next;
        head.pre = null;
        return val;
    }

获取队头元素

获取队头元素则十分的简单,使用个临时变量拿到头节点的值然后返回就可以了

    //拿出队列的首个元素进行查看
    public int peek() {
        if (head == null) {
            throw new ExceptionOfEmpty("队列为空,无法操作");
        }
        int val = head.val;
        return val;
    }

获取队列长度

获取队列的长度也非常的简单,使用一个新的节点挨个变量,然后累加计数器最后返回就可以

    //获取队列长度
    public int size() {
        int count = 0;
        ListNode cur = head;
        while (head != null) {
            cur = cur.next;
            count++;
        }
        return count;
    }

循环队列的概念

循环队列是一种特殊的队列数据结构,它允许队列的头尾相接,形成一个环。循环队列解决了普通队列的一些缺点,如空间浪费和在元素出队后无法再次入队的问题。

循环队列通常使用数组来实现,其内部有两个指针frontrear,分别指向队列的第一个元素和最后一个元素的下一个位置。初始化时,frontrear都指向数组的第一个位置。

当往循环队列中插入元素时,将元素放入rear指向的位置,并将rear指针向后移动一位。如果rear指针到达数组的尾部,则将其指向数组的起始位置。当从循环队列中删除元素时,将front指针向后移动一位,表示该元素已出队列。如果front指针到达数组的尾部,则将其指向数组的起始位置。

图示如下:

使用循环队列可以实现高效的元素入队和出队操作,因为循环队列的空间利用率较高。当队列满时,rear指针和front指针会指向同一个位置,此时队列被认为是满的。而在普通队列中,即使队列中还有空闲位置,当rear指针到达数组的尾部时,无法再向队列中插入元素。

循环队列通过循环利用数组空间,解决了普通队列的空间浪费和无法再次入队的问题,提高了队列的空间利用率和性能。


循环队列的数据结构

对于循环队列,我们可以使用数组来模拟实现,分别有俩个指针指向整个数组的首部和尾部

public class MyCircularQueue {
    public int[] elem;//存放数据
    public int front;//指向队头
    public int rear;//指向队尾
    
    MyCircularQueue(int k) {
        elem = new int[k];
    }
}

循环队列的实现

对于循环队列的实现,其实难点就在于判断队列的状态,分得清队列已满和队列为空的情况就可以,其余的操作实际上都是非常简单的数组的添加和删除元素

另外很重要的一点就是,循环队列需要体现出循环的概念,所以不管是队头指针还是队尾指针,我们在操作的时候都需要除以这个数组的模

判断队列是否为空

队列为空是怎么样的状态?为空,也就是说队列内什么都没有,队列的开始就是队列的结束,也就是队头指针和队尾指针指向同一个地方,如图所示:

    //判空 front和rear相遇
    public boolean isEmpty() {
        return front == rear;
    }

判断队列是否已满

队列已满是怎么样的状态?队列已满就是说队列每一个位置都有元素存放,这个情况下,如果队尾指针再往后走一步,就相当于回到了第二轮的起点,如图所示:

那么也就是说,当队尾指针再往后走一步的值除以整个队列的大小,就等于头指针的大小 

    //判空 front和rear相遇
    public boolean isFull() {
        return (rear + 1) % elem.length == front;
    }

入队

入队的时候,我们先要判断队列是否已满,满了就返回false表示入队失败;没有满的话就进行入队,也就是从数组的最后位置放个元素,然后再更改队尾指针指向,图示如下:

出队

出队的实现实际上就是对数组的第一个元素进行删除操作,如图所示:

    //出队
    public boolean deQueue() {
        if (isEmpty()) {
            return false;
        }
        front = (front + 1) % elem.length;
        return true;
    }

得到队头元素

这样的操作其实就是出队操作的简化,拿到值返回即可,这里就不再赘述

    //得到队头元素
    public int Front() {
        if (isEmpty()) {
            return -1;
        }
        return elem[front];
    }

得到队尾元素

这里有一点需要注意,当我们的队列只有一个元素的时候,是需要进行特殊处理的,不然就会出现数组越界的异常,毕竟数组不存在下标为-1的元素

    //得到队尾元素
    public int Rear() {
        if (isEmpty()) {
            return -1;
        }
        int index = (rear == 0) ? elem.length - 1 : rear - 1;
        return index;
    }



  本次的分享就到此为止了,希望我的分享能给您带来帮助,也欢迎大家三连支持,你们的点赞就是博主更新最大的动力!如有不同意见,欢迎评论区积极讨论交流,让我们一起学习进步!有相关问题也可以私信博主,评论区和私信都会认真查看的,我们下次再见

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【网络安全】-Linux操作系统—VMWare软件

文章目录 VMWare软件的安装选择VMWare版本下载VMWare安装过程 VMWare的常用操作创建新的虚拟机配置虚拟机启动和关闭虚拟机安装VMWare Tools VMWare的克隆和快照克隆(Clone)快照(Snapshot) 总结 VMWare是一种流行的虚拟化软件&…

物联网对接使用蓝牙还是WiFi,应该如何选择?

蓝牙是一种无线技术协议,可促进连接设备之间短距离的数据交换。它依赖于物理邻近性并使用2.400至2.485 GHz之间的UHF(超高频)无线电波。蓝牙旨在创建个人区域网络(PAN)并在笔记本电脑、智能手机和外围设备等计算设备之…

webview 的 title 和 url

在Appium以混合型App进行自动化操作时,遇到WebView时切换至WebView才能进行操作。当遇到多个WebView时,可以利用 title 和 url 切换至相应的 WebView。

✺ch5——纹理贴图

目录 加载纹理图像文件纹理坐标在着色器中使用纹理:采样器变量和纹理单元纹理贴图:示例程序多级渐远纹理贴图各向异性过滤环绕和平铺透视变形材质——更多OpenGL细节补充说明 纹理贴图是在栅格化的模型表面上覆盖图像的技术。 它是为渲染场景添加真实感的…

宏基因组学Metagenome-磷循环Pcycle功能基因分析-从分析过程到代码及结果演示-超详细保姆级流程

大背景介绍 生信分析,凡事先看论文,有了论文就有了参考,后续分析就有底了,直接上硬菜开干: PCycDB: a comprehensive and accurate database for fast analysis of phosphorus cycling genes - PubMed 数据库及部分分析代码github库: GitHub - ZengJiaxiong/Phospho…

docker 与 ffmpeg

创建容器 docker run -it -v /mnt/f/ffmpeg:/mnt/f/ffmpeg --name ffmpeg 49a981f2b85f /bin/bash 在 Linux 上编译 FFmpeg: 安装依赖库: sudo apt-get update sudo apt-get install build-essential yasm cmake libtool libc6 libc6-dev unzip wget下…

SpringCloud-高级篇(八)

(1)TCC模式 前面学了XA和AT模式,这两种模式最终都能实现一致性,和隔离性,XA是强一致,AT是最终一致,隔离性呢XA是在第一阶段不提交,基于事务本身的特性来完成隔离,AT则是…

Python都有什么特性,为什么适合每个人学习?详细解读来了

Python编程语言简介 Python,一种高层次的、解释型的编程语言,以其跨平台性、易读性和灵活性在全球编程界占据了重要的地位。自1991年首次发布以来,Python已经成为了无数程序员和企业的首选语言,尤其是在快速开发和原型设计方面。 …

Ansible:模块1

Ansible: 远程操作主机功能 自动化运维(playbook 剧本 yaml) 是基于python开发的配置管理和应用部署工具。在自动化运维中,现在是一军突起。 Ansible能批量配置,部署,管理上千台主机。类似于xshell的一…

MFC 程序执行流程

目录 MFC 程序启动 MFC 入口函数 程序执行流程总结 在Win32课程中WinMain由程序员自己实现,那么流程是程序员安排,但到了MFC中,由于MFC库实现WinMain,也就意味着MFC负责安排程序的流程。 MFC 程序启动 程序的启动,…

红枣期货个股(红枣期货个股:投资方向分析)

红枣期货个股介绍 红枣是我国传统的绿色健康食品,具有营养丰富、味道独特的特点,深受消费者喜爱。红枣产业链较长,包括种植、采摘、加工、销售等环节,其中期货是红枣产业不可或缺的一环。红枣期货个股作为期货交易市场上的重要投…

mybatis中oracle的sql没走索引导致特别慢(未加jdbcType的)

如果直接跑sql是能走索引很快,在mybatis中不能,可能就是jdbcType的原因。 比如,我有一个属性A,在表里面是VARCHAR2类型,但是在mybatis中的sql是#{a},缺少jdbcTypeJdbcType.VARCHAR,就会导致myba…

risc-v system instruction

ECALL ecall 指令以前叫做 scall,用于执行环境的变更,它会根据当前所处模式触发不同的执行环境切换异常, 用来执行需要更高权限才能执行的功能;简单来说,ecall 指令将权限提升到内核模式并将程序跳转到指定的地址。操作系统内核和应用程序其实…

AD采集卡设计方案:630-基于PCIe的高速模拟AD采集卡

基于PCIe的高速模拟AD采集卡 一、产品概述 基于PCIe的一款分布式高速数据采集系统,实现多路AD的数据采集,并通过PCIe传输到存储计算服务器,实现信号的分析、存储。 北京太速科技,产品固化FPGA逻辑,适配2路…

ShardingSphere-JDBC 和 ShardingSphere-Proxy,你选择哪一个

参考文章 总结: 只使用Java,ShardingSphere-JDBC更好有异构语言的话,ShardingSphere-Proxy 更好混用也挺香

Flink系列之:监控Checkpoint

Flink系列之:监控Checkpoint 一、概览二、概览(Overview)选项卡三、历史记录(History)选项卡四、历史记录数量配置五、摘要信息(Summary)选项卡六、配置信息(Configuration&#xff…

100GPTS计划-AI动漫AnimeArtisan

地址 https://poe.com/AnimeArtisan https://chat.openai.com/g/g-LM6ObVhfF-anime-artisan 测试 风景类: 阳光、蓝天、白云、大海、海滩、森林、瀑布、山峰、雪山 日常类: 睡觉、跑步、学习、工作、做家务、看书、听音乐、运动、购物、煮饭 人物类: 女孩、男孩、老人、儿童…

『 Linux 』重新理解挂起状态

文章目录 🦄 前言新建状态 🐋挂起状态 🐋唤入唤出 🐋进程与操作系统间的联系 🐋 🦄 前言 『 Linux 』使用fork函数创建进程与进程状态的查看中提到了对挂起状态的一个理解; ​ 挂起状态相比于其…

爬虫练习-获取imooc课程目录

代码: from bs4 import BeautifulSoup import requests headers{ User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:94.0) Gecko/20100101 Firefox/94.0, }id371 #课程id htmlrequests.get(https://coding.imooc.com/class/chapter/id.html#Anchor,head…

kubernetes 学习笔记

1. Kubernetes 介绍 1.1 应用部署方式的演变 在部署应用程序的方式上,主要经理了三个时代: 传统部署:互联网早期,会直接将应用程序部署在物理机上。虚拟化部署:可以在一台物理机上运行多个虚拟机,每个虚…
最新文章