【数据结构】队列的实现与优化指南

news2025/1/22 19:13:42

一、前言

队列是一种重要的数据结构,它按照“先入先出”(FIFO)的原则管理数据。本文将介绍队列的概念、应用场景,以及如何使用数组实现普通队列和环形队列。


二、内容

2.1 概述

(1)什么是队列?

队列(Queue)是一种常见的数据结构,它是一个线性数据结构,按照先入先出(FIFO,First-In-First-Out)的原则来管理数据。

注意,先入先出的原则就意味着最早进入队列的元素将最先被取出,而最后进入队列的元素将最后被取出,类似于排队等候服务的行为。

队列可以使用数组或链表来实现,具体实现方式因应用需求而异。

队列支持两种主要的操作,即入队(Enqueue)和出队(Dequeue)。

  • 入队:将元素添加到队列的尾部。
  • 出队:从队列的头部取出元素并删除它。

(2)应用场景

队列的应用场景有很多,比如:

  1. 任务调度:操作系统使用队列来管理待执行的任务或进程,确保按照进入队列的顺序分配处理时间。
  2. 数据缓冲:队列用于数据传输和处理中,特别是在异步通信或生产者-消费者模式中,可以缓冲待处理的数据。
  3. 广度优先搜索:在图论和搜索算法中,队列用于实现广度优先搜索,以逐层遍历图结构。
  4. 打印任务队列:打印机队列用于管理待打印的文档,确保按照顺序打印。
  5. 网页请求队列:Web服务器可以使用队列来处理收到的请求,以便有序响应客户端请求。
  6. 排队系统:在银行、餐馆、医院等场所,队列被用来管理等待服务的客户,确保服务按照先来先服务的原则。
  7. ......

队列在计算机科学和实际应用中非常有用,因为它们提供了一种有效的方法来管理和调度数据或任务,以确保按照特定的顺序进行处理。


2.2 数组模拟队列

下面,我们用数组来模拟一个简单的队列数据结构。

(1)队列类定义

首先给出类的定义:

class ArrayQueue {
    private int maxSize;
    private int front;
    private int rear;
    private int[] data;
    
    ArrayQueue(int queueMaxSize) {
        maxSize = queueMaxSize;    // 队列的最大容量
        data = new int[maxSize];    // 存放队列的数据
        front = -1;    // 指向队列头的前一个位置
        rear = -1;     // 直接指向队列尾部
    }
	
    // ... 方法定义
}

在这里,ArrayQueue 是一个队列类,使用数组作为内部数据存储。它包括最大容量(maxSize)、队列头(front)、队列尾(rear)和一个整数数组(data)来存放队列的数据。

构造函数 ArrayQueue 接受一个整数参数 queueMaxSize,表示队列的最大容量。初始化时,队列的头(front)和尾都(rear)被置为-1,表示队列为空。

需要注意这里的定义,在这里,front 变量指的是指向队列首元素的前一个位置,而 rear 变量则指向队列的尾部元素,即最后一个元素。

因此,初始队列的结构图如下:

image-20231016142126890.png

(2)isEmpty

public boolean isEmpty() {
    return rear == front;
}

isEmpty() 方法用于检查队列是否为空。如果队列头和队列尾相等,表示队列中没有数据,返回 true;否则返回 false

(3)isFull

public boolean isFull() {
    return rear == maxSize - 1;
}

isFull() 方法用于检查队列是否已满。如果队列尾等于最大容量减1,表示队列已满,返回 true;否则返回 false

(4)enQueue

// 入队操作,添加数据到队尾
public void enQueue(int num) {
    if(isFull()) {
        System.out.println("队列已满,无法入队");
        return;
    }
    rear++;
    data[rear] = num;
}

enQueue 方法用于将数据添加到队列的尾部。首先,它会检查队列是否已满,如果是,将输出一条错误消息并不执行入队操作。如果队列未满,将 rear 后移,然后将数据存入队列尾部。

再次强调一下,这里的 rear 变量用于指向队列的最后一个数据,即队列的尾部。

image-20231016142544266.png

(5)deQueue

// 出队操作,取出队头数据
public int deQueue() {
    if(isEmpty()) {
        throw new RuntimeException("队列为空,无法出队"); 
    }
    front++;
    return data[front];
}

deQueue 方法用于取出队列头部的数据。首先,它会检查队列是否为空,如果是,将抛出一个运行时异常。如果队列不为空,将 front 后移,然后返回队头的数据。

再次强调一下,这里的 front 变量指向的是队列头数据的前一个位置。

image-20231016142948770.png

(6)headQueue

// 查看队头数据(注意不是取出数据)
public int headQueue() {
    if(isEmpty()) {
        throw new RuntimeException("队列为空,没有数据");
    }
    return data[front+1];
}

headQueue 方法用于获取队列头部的数据,但不会将其出队。它会检查队列是否为空,如果是,将抛出一个运行时异常。如果队列不为空,将返回队头的数据。

(7)showQueue

// 打印队列
public void showQueue() {
    if(isEmpty()) {
        System.out.println("队列为空,没有数据");
        return;
    }
    // 简单的遍历队列
    for(int i = 0; i < data.length; i++) {
        System.out.printf("data[%d] = %d\n", i, data[i]);
    }
}

showQueue 方法用于简单地打印队列的所有元素。如果队列为空,将输出一条消息表示队列为空。否则,它会简单地遍历队列,打印每个数据元素的索引和值。


2.3 数组模拟环形队列

(1)存在的问题

我们再来思考一个问题,虽然上述的队列类实现了一个简单的队列数据结构,但仍然存在弊端。那就是数组使用一次后不能复用

什么意思?

具体的,我们可以发现,每当队列入队一个数据,rear 变量就会往后移一位。每当有元素出队,front 变量也会往后移一位。但是!一旦 rear 变量到达队列的尾部,如果队列头部仍有空余的空间,就像这样:

image.png

那么此时根据 isFull() 方法的判断下,该队列是满的。因此无法再入队。

因此我们说,对于之前的队列简单实现来说,一旦队列中的数据元素被取出,对应的数组位置就不能再次使用。数据从头部添加,从尾部取出。一旦数组被填满,我们无法再添加新的数据,即使队列的前面已经有一些位置被释放出来。这就会导致内存资源浪费

为了解决这个问题,我们考虑使用环形队列来优化。

什么是环形队列

事实上,环形队列是一种更高效的队列实现方式,它允许队列在达到最大容量后继续添加元素,以覆盖掉队列头部已经被取出的数据,实现数据的循环复用。

我们通过取模运算 % 来实现环形队列。

(2)思路分析

当我们考虑了队列内部数据存储资源的复用后,我们就需要对 front 和 rear 变量的含义进行一个的调整(当然不调整也行,看个人习惯)。

具体如下:

  • front 变量:
    • 表示指向队列的第一个元素,即首元素。
    • data[front] 是队列的第一个元素。
    • front的初始值为 0。
  • rear 变量:
    • 表示指向队列最后一个元素的下一个位置
    • 这是为了表示队列中哪些位置是可用的,以便继续添加新的元素。
    • rear 的初始值同样为 0。

当我们这样约定好了后,就可以开始着手编写代码,得到一个环形队列。

此时判断队列已满或空时,逻辑需要略微调整。

判断环形队列空时,条件为:(rear == front)。因为当 rear 指针等于 front 指针时,表示队列没有有效的元素,即队列为空。

判断环形队列满时,条件为:(rear + 1) % maxSize == front

这该怎么理解?

事实上,在含义调整后,环形队列中的 rear 变量指向的位置实际上就是预留给下次入队的数据存放的位置

当有一个新的数据入队时,rear 指向的位置就可以存储本次入队的数据的值,然后,rear 就会加一并取余 maxSize ,用于寻找下一个可以存储入队数据的位置。

因此,当(rear + 1) % maxSize 的值刚好等于 front,那么证明该环形队列已经满了,没有地方可以存储下一次入队的值。

举一个例子,假设 maxSize 为 3,初始时 front 和 rear 都是0:

  • 队列为空:front = 0rear = 0
  • 插入一个元素:front = 0rear = 1
  • 插入第二个元素:front = 0rear = 2
  • 插入第三个元素:front = 0rear = 0(此时队列满,因为 (rear + 1) % maxSize 等于 front
  • 取出第一个元素:front = 1rear = 0(此时队列有效元素个数为 2,因为 (0+3-1) % 3 == 2

示意图如下:

image-20231016153249708.png

(3)优化后的队列类

优化后的代码实现如下:

class CircleArrayQueue {
    private int maxSize;
    private int front;    // 初始值为 0,指向队头数据,即首元素
    private int rear;     // 初始值为 0,指向队尾数据的下一个位置
    private int[] data;
	
    ArrayQueue(int queueMaxSize) {
        maxSize = queueMaxSize;	
        data = new int[maxSize];
    }
	
    // 判断队列是否为空
    public boolean isEmpty() {
        return rear == front;
    }
	
    // 判断队列是否满
    public boolean isFull() {
        return (rear + 1) % maxSize == front;
    }
	
    // 入队:添加数据到队尾
    public void enQueue(int num) {
        if(isFull()) {
            System.out.println("队列已满,无法入队");
            return;
        }
        data[rear] = num;
        rear = (rear + 1) % maxSize;
    }
	
    // 出队,取出队头数据
    public int deQueue() {
        if(isEmpty()) {
            throw new RuntimeException("队列为空,无法出队"); 
        }
        int value = data[front];
        front = (front + 1) % maxSize;
        return value;
    }
	
    // 显示队列的头数据(不是取出数据)
    public int headQueue() {
        if(isEmpty()) {
            throw new RuntimeException("队列为空,没有数据");
        }
        return data[front];
    }
	
    // 返回环形队列当前的元素个数
    public int size() {
        return (rear + maxSize - front) % maxSize;
    }
	
    // 打印队列
    public void showQueue() {
        if(isEmpty()) {
            System.out.println("队列为空,没有数据");
            return;
        }
        // 遍历思路,从 data[front] 遍历到 data[front + size]
        for(int i = front; i < front + size(); i++) {
            System.out.printf("data[%d] = %d\n", i%maxSize, data[i%maxSize]);
        }
    }
}

三、总结

本文详细介绍了队列数据结构的概念和应用,包括普通队列和环形队列的实现。队列是一种有序的数据结构,它在计算机科学中被广泛应用,用于管理数据和任务的顺序执行。普通队列使用数组实现,但存在内存资源浪费的问题。为了解决这个问题,我们引入了环形队列的概念,它允许队列数据的循环复用,更加高效地利用内存。

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

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

相关文章

ant的echo任务

通过echo&#xff0c;可以将一个消息输出到当前的日志记录器、监听器&#xff0c;如果没有覆盖的话&#xff0c;默认就是在控制台输出&#xff08;即System.out&#xff09;。 也可以将消息发送到文件。 可以使用的参数&#xff1a; 举例&#xff0c;build.xml文件的的内容如…

Sectigo OV通配符1590元

通配符SSL证书是一种特殊的SSL证书&#xff0c;它能够为多个域名提供加密保护&#xff0c;这种证书可以用于保护一个主域名及其所有子域名&#xff0c;适合子域名比较多的个人或者企事业单位使用。通配符SSL证书既节省了管理证书的时间&#xff0c;又减少了购买SSL证书的成本&a…

7、Linux驱动开发:设备-自动创建设备节点

目录 &#x1f345;点击这里查看所有博文 随着自己工作的进行&#xff0c;接触到的技术栈也越来越多。给我一个很直观的感受就是&#xff0c;某一项技术/经验在刚开始接触的时候都记得很清楚。往往过了几个月都会忘记的差不多了&#xff0c;只有经常会用到的东西才有可能真正记…

【数字人】4、AD-NeRF | 使用 NeRF 来实现从声音到数字人人脸的直接驱动(ICCV2021)

文章目录 一、背景二、方法2.1 适用于 talking head 的神经辐射场2.2 使用辐射场来进行体渲染2.3 独立 NeRF 表达 三、效果 论文&#xff1a;AD-NeRF: Audio Driven Neural Radiance Fields for Talking Head Synthesis 代码&#xff1a;https://github.com/YudongGuo/AD-NeRF…

​蔚来 V2G:带你重新了解如何参与构建新型电力系统

8 月 23 号&#xff0c;蔚来低调的在官方社区社区内发布了一条推文&#xff0c;推文为 「蔚来参与全国最大规模 V2G 需求响应」。 而车网互动验证中心(简称 e-Park)的 V2G 试验&#xff0c;在江苏无锡正式启动。 本次响应时间为 30 分钟&#xff0c;50 台电动车送电近 2000kW…

25.1 MySQL SELECT语句

1. SQL概述 1.1 SQL背景知识 1946年, 世界上诞生了第一台电脑, 而今借由这台电脑的发展, 互联网已经成为一个独立的世界. 在过去几十年里, 许多技术和产业在互联网的舞台上兴衰交替. 然而, 有一门技术却从未消失, 甚至日益强大, 那就是SQL.SQL(Structured Query Language&…

【软件工程】简单讲讲设计模式七大原则,以及代码简单举例

给自己一个目标&#xff0c;然后坚持一段时间&#xff0c;总会有收获和感悟&#xff01; 学软件或计算机专业的同学应该都会接触到一门课程《软件工程》&#xff0c;七大设计原则属于软件工程中的重要知识点。 目录 一、软件工程1.1、提供指导和规范1.2、确保软件质量1.3、提高…

【经验分享】解决vscode编码问题

目录 先看一下我遇到的问题和你们的一不一样 下面是我查到的解决办法&#xff1a; 简单点说就是 我们看看解决后的效果 先看一下我遇到的问题和你们的一不一样 我一开始以为就是编码问题。 下面是我查到的解决办法&#xff1a; 这个错误提示看起来仍然是中文乱码。可能是由于…

Flink学习之旅:(二)构建Flink demo工程并提交到集群执行

1.创建Maven工程 在idea中创建一个 名为 MyFlinkFirst 工程 2.配置pom.xml <properties><flink.version>1.13.0</flink.version><java.version>1.8</java.version><scala.binary.version>2.12</scala.binary.version><slf4j.ver…

Java构建Web项目

对无底线服务型的系统&#xff0c;业务代码和界面代码脚本化是及其重要的。一是脚本化能确保部署本地就是再用的代码&#xff0c;不存在为每个项目管理代码的问题。然后脚本化不需要人为编译和投放程序库。极大的简化维护难度和成本。能不能脚本化直接决定了能否全面铺开运维&a…

流量新玩法:微信问一问了解一下

来自一位不断探索的营销人的问题&#xff1a;微信“问一问”引流&#xff0c;一个问答引流1000精准粉&#xff0c;是不是真的&#xff1f;如果是真的&#xff0c;那该怎么做呢&#xff1f; 微信的问一问功能&#xff0c;支持图文回答&#xff0c;也支持用视频去回答&#xff0c…

Java使用javah命令:‘javah‘ 不是内部或外部命令,也不是可运行的程序或批处理文件。

前提是已安装jdk&#xff0c;配置JDK环境变量&#xff0c;并成功输出下图&#xff1a; 但在命令行窗口使用javah&#xff0c;找不到该命令&#xff1a; 原因&#xff1a;新版的Java不使用javah的命令了&#xff0c;而是使用javac -h 用法&#xff1a; javac -h <directory&…

低代码系列——可视化编辑器

前端社区里&#xff0c;低代码/无代码是被讨论的火热赛道。它通过用最少量的编程代码去开发应用程序&#xff0c;从而提高效率。由此&#xff0c;许多企业都在使用低代码平台进行业务的开发和升级。低代码平台可以大幅简化编码过程&#xff0c;并且可以快速构建定制化的应用程序…

C++ - 类型转换 - static_cast - reinterpret_cast - const_cast - dynamic_cast

目录 类型转换 C语言当中的类型转换 为什么C需要四种类型转换 &#xff08;讲解volatile关键字&#xff09; C强制类型转换 static_cast reinterpret_cast const_cast dynamic_cast&#xff08;动态转换&#xff09; RTTI 类型转换 C语言当中的类型转换 其实在 C语言当…

从理论到实践,实时湖仓功能架构设计与落地实战

在上篇文章中&#xff0c;我们向大家解释了为什么实时湖仓是当前企业数字化转型过程中的解决之道&#xff0c;介绍了实时计算和数据湖结合的应用场景。&#xff08;“数据驱动”时代&#xff0c;企业为什么需要实时湖仓&#xff1f;&#xff09; 在这篇文章中&#xff0c;我们…

使用Gitlab构建简单流水线CI/CD

什么是Gitlab Gitlab实质上是一套DevOps工具 目前看起来&#xff0c;Gitlab属于是内嵌了一套CI/CD的框架&#xff0c;并且可以提供软件开发中的版本管理、项目管理等等其他功能。 这里需要辨别一下Gitlab和Github Gitee的区别。 GIthub大家都很熟悉了&#xff0c;一般大家都会…

探索DeFi世界,MixGPT引领智能金融新时代

随着区块链技术的迅猛发展&#xff0c;DeFi&#xff08;去中心化金融&#xff09;正成为金融领域的新宠。在这个充满活力的领域里&#xff0c;MixTrust站在创新的前沿&#xff0c;推出了一款引领智能金融新时代的核心技术——MixGPT。 MixGPT&#xff1a;引领智能金融体验的大型…

Rust逆向学习 (1)

文章目录 Hello, Rust Reverse0x01. main函数定位0x02. main函数分析line 1line 2line 3line 4~9 0x03. IDA反汇编0x04. 总结 近年来&#xff0c;Rust语言的热度越来越高&#xff0c;很多人都对Rust优雅的代码和优秀的安全性赞不绝口。对于开发是如此&#xff0c;对于CTF也是如…

208. 开关问题 - 异或方程组

208. 开关问题 - AcWing题库 我们可以找每一个开关由哪些开关掌控&#xff0c;每一个开关的值设为动过为1&#xff0c;没动过为0 再看当前开关的状态与结果的状态是否一致&#xff0c;一致为0&#xff0c;说明掌控这个开关的开关门的异或值为0&#xff0c;不一致则为1&#xf…

彻底理解操作系统与内核的区别!

通用底盘技术 Canoo公司有一项核心技术专利&#xff0c;这就是它们的通用电动底盘技术&#xff0c;长得是这个样子&#xff0c;非常像一个滑板&#xff1a; 这个带轮子、有电池、能动的滑板已经包含了一辆车最核心的组件&#xff0c;差的就是一个外壳。这个看起来像滑板的东西…