【数据结构】从数据结构角度深入探究队列

news2025/1/12 6:00:33

队列是计算机科学中的一种基本数据结构,用于存储和管理数据。在计算机程序中,队列被广泛应用于任务调度、进程管理等场景。本文将介绍队列的概念、特点、常见操作以及应用。

文章目录

  • 队列的概念
  • 队列的应用
  • 队列的存储结构
  • 队列接口的实现
    • 队列的初始化
    • 队尾入队列
    • 队头出队列
    • 获取队列头部元素
    • 获取队列队尾元素
    • 获取队列中有效元素个数
    • 检测队列是否为空
    • 销毁队列
  • 总结

队列的概念

你们在用电脑时有没有经历过,机器有时会处于疑似死机的状态,鼠标点什么似乎都没用,双击任何快捷方式都不动弹。就当你失去耐心,打算reset时,突然它像酒醒了一样,把你刚才单击的所有操作全部都按顺序执行了一遍,这是因为操作系统在当时可能CPU一时忙不过来,等前面的事忙完后,后面多个指令需要通过一个通道输出,按先后次序排队执行造成的结果。所以在这操作系统中,是应用了一种数据结构来实现刚才提到的先进先出的排队功能,这就是队列。

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

  • 队列的特点

队列具有以下几个特点:

  1. 先进先出。最早进入队列的元素在队列中排在最前面,最后进入队列的元素排在最后面。
  2. 只能在队列的前端执行出队操作,在队列的后端执行入队操作。

在这里插入图片描述


队列的应用

队列在计算机程序中有很多应用场景,下面介绍其中几个常见的应用。

  1. 消息传递:
    在消息传递模型中,消息被发送到队列中等待接收方进行处理。发送方可以通过入队操作向队列发送消息,而接收方则通过出队操作从队列中获取消息。
  2. 缓存
    在网络通信中,缓存被广泛应用于加速数据传输。当一个请求到达时,它可能需要从远程服务器中获取数据,这个过程需要等待一定时间。为了避免每次请求都需要重新获取数据,可以将数据缓存在本地队列中。当下一次请求到达时,可以直接从队列中获取数据,避免了等待时间。
  3. 任务调度
    在许多系统中,任务必须按照一定的顺序进行处理。例如,操作系统中的进程调度就是一种任务调度。当一个进程被创建时,它会被添加到系统的任务队列中等待处理。调度器通过出队操作从队列中选择下一个要执行的进程,并将其分配给CPU执行。
  • 总的来说队列是计算机科学中非常重要的一种数据结构,具有先进先出的特点。队列通过入队和出队操作管理数据,常见的应用场景包括任务调度、消息传递和缓存等。对于程序员来说,了解队列的基本概念和常见操作是非常必要的。

队列的存储结构

队列可以使用数组或者链表的结构来实现,那哪一个存储结构更加优呢?下面我们来分析他们在队列中存储各自的优缺点。

  • 数组存储结构的优点:
    1 . 数组在内存中是连续存储的,因此访问元素时速度较快。CPU高速缓存命中率会更高。
    2 . 数组实现相对简单,不需要额外的指针来维护元素之间的关系。
  • 数组存储结构的缺点
    1 . 需要事先确定队列的最大长度,这可能会导致性能下降。
    2 . 需要移动元素来保持队列的顺序.

  • 链式存储的优点
    1 . 不需要事先确定队列的最大长度,可以动态扩展.
    2 . 插入和删除操作只需要修改指针,不需要移动元素.
    3 . 可以实现多个队列共享一个链表
  • 链式存储的缺点
    1 . CPU高速缓存命中率会更低,不是连续存储的,因此访问元素时速度较慢。
    2 .实现相对复杂。

  • 终言: 使用链表的结构实现更优一些,因为数据存储有一个致命的缺点就是大量挪动元素。

队列的链式存储结构,其实就是线性表的单链表,只不过它只能尾进头出而已,我们把它简称为链队列。为了操作上的方便,我们将队头指针head指向链队列的头结点,而队尾tail尾指针指向终端结点.。

// 定义队列节点结构体
typedef int QDataType;
typedef struct QueueNode
{
    struct QueueNode* next; // 指向下一个节点的指针
    QDataType data; // 节点数据
}QNode;

// 定义队列结构体
typedef struct Queue
{
    QNode* phead; // 队列头指针,指向第一个节点
    QNode* ptail; // 队列尾指针,指向最后一个节点
    int size; // 队列大小,即节点数量
}Queue;

在这个链表中,每个节点包含两个成员变量:指向下一个节点的指针和节点数据。队列结构体中包含头指针、尾指针和节点数量三个成员变量。头指针指向队列的第一个节点,尾指针指向队列的最后一个节点,节点数量表示队列中节点的数量。由于是单向链表,所以只需要维护头指针和尾指针就可以实现入队和出队操作。


队列接口的实现

队列的初始化

  • 先把队列元素全部置空,size置0.
void QueueInit(Queue* pq)
{
	assert(pq);

	pq->phead = NULL;
	pq->ptail = NULL;
	pq->size = 0;
}

队尾入队列

在这里插入图片描述

  • 详细步骤已放代码注释,以下是该入队列核心思路.
    1. 如果队列为空,将头指针和尾指针都指向新节点。
    2. 如果队列不为空,将尾指针指向新节点,并将前一个节点的指向下一个节点的指针更新为新节点。
void QueuePush(Queue* pq, QDataType x)
{
    assert(pq); // 断言:如果传入的队列 pq 为空,程序将停止运行并提示错误。
    
    // 创建新节点 newnode 并为其分配内存空间。
    QNode* newnode = (QNode*)malloc(sizeof(QNode));
    if (newnode == NULL)
    {
        perror("malloc fail\n"); // 如果分配空间失败,则程序将输出一条错误信息。
        return;
    }

    // 设置新节点的数据和指针值。
    newnode->data = x; // 新节点中储存的数据值为变量 x。
    newnode->next = NULL; // 新节点的指向下一个节点的指针初始值设为 NULL。

    // 将新节点加入队列末尾。
    if (pq->ptail == NULL) // 如果队列为空(没有节点),则头指针和尾指针均指向新节点。
    {
        assert(pq->phead == NULL); // 确认队首指针也为 NULL。

        pq->phead = pq->ptail = newnode;
    }
    else // 否则,仅将尾指针指向新节点,并将前一个节点的指向下一个节点的指针更新为新节点。
    {
        pq->ptail->next = newnode;
        pq->ptail = newnode;
    }

    pq->size++; // 队列长度加一。
}


队头出队列

  • 详细步骤已放代码注释,以下是该入队列核心思路.
    1 . 出队列前要判断该队列是否为空.
    2 . 如果队列中只有一个节点,即头指针和尾指针指向同一个节点,则释放头结点.并将将头指针和尾指针均置为NULL.
    3 . 如果队列中有不止一个节点,即头指针和尾指针指向不同的节点,则保存头结点的下一个节点,释放头结点,将头指针指向下一个节点.
void QueuePop(Queue* pq)
{
    assert(pq);  // 判断参数pq是否为空
    assert(!QueueEmpty(pq));  // 判断队列是否为空

    if (pq->phead->next == NULL)  // 如果队列中只有一个节点
    {
        free(pq->phead);  // 释放头结点
        pq->phead = pq->ptail = NULL;  // 将头指针和尾指针均置为NULL
    }
    else  // 如果队列中有不止一个节点
    {
        QNode* next = pq->phead->next;  // 保存头结点的下一个节点
        free(pq->phead);  // 释放头结点
        pq->phead = next;  // 将头指针指向下一个节点
    }

    pq->size--;  // 减小队列的大小
}

获取队列头部元素

  • 需要注意的是要判断该队列是否为空.如不为空,则返回队列头节点的数据域。
QDataType QueueFront(Queue* pq)
{
    assert(pq);  // 判断参数pq是否为空
    assert(!QueueEmpty(pq));  // 判断队列是否为空

    return pq->phead->data;  // 返回头结点所指向节点的数据值
}

获取队列队尾元素

  • 需要注意的是要判断该队列是否为空…如不为空,则返回队列尾节点的数据域。
QDataType QueueBack(Queue* pq)
{
    assert(pq);  // 判断参数pq是否为空
    assert(!QueueEmpty(pq));  // 判断队列是否为空

    return pq->ptail->data;  // 返回尾结点所指向节点的数据值
}

获取队列中有效元素个数

  • 返回队列的成员变量size的值即可.
int QueueSize(Queue* pq)
{
    assert(pq);  // 判断参数pq是否为空

    return pq->size;  // 返回队列的成员变量size的值
}

检测队列是否为空

  • 判断队列的成员变量size是否为0,如果为0则说明队列为空,返回true,否则返回false。
bool QueueEmpty(Queue* pq)
{
    assert(pq);  // 判断参数pq是否为空

    return pq->size == 0;  // 如果队列为空,返回true,否则返回false
}

销毁队列

  • 详细步骤已放代码注释,需要注意的是每次销毁节点时,要保存它下一个节点。
void QueueDestroy(Queue* pq)
{
	assert(pq);  // 判断参数pq是否为空

	QNode* cur = pq->phead;  // 定义一个指针cur指向队列头结点pq->phead。
	while (cur)  // 利用while循环遍历链表
	{
		QNode* next = cur->next;  // 定义一个指针next指向当前节点的下一个节点。
		free(cur);  // 释放当前节点的内存。
		cur = next;  // 将cur指向next节点。
	}

	pq->phead = pq->ptail = NULL;  // 将队列头结点和尾结点置为NULL。
	pq->size = 0;  // 将队列大小设置为0。
}

总结

  • 链式队列不需要移动元素,因此相比于顺序队列,链式队列的空间利用率更高。. 链式队列的缺点是需要额外的空间存储指针,因此相比于顺序队列,链式队列的空间复杂度更高。总之,链式队列是一种高效、灵活的队列数据结构,适用于需要频繁进行入队和出队操作的场景。

  • 队列是一种简单但非常有用的数据结构,掌握它的基本概念、操作和实现方式对于编程学习和刷题,后面的面试都非常重要。

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

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

相关文章

plt.plot(x,y,color,linestyle,marker)函数参数详解

一、plt.plot()函数详解 plt.plot()函数是matplotlib库中用于绘制线条图的函数,它有多个参数可以控制绘图的各个方面。以下是常用的一些参数及其作用: x: x轴数据的列表或数组y: y轴数据的列表或数组linewidth: 线条的宽度,从0到无穷大的浮…

( 动态规划) 115. 不同的子序列 ——【Leetcode每日一题】

❓115. 不同的子序列 难度:困难 给你两个字符串 s 和 t ,统计并返回在 s 的 子序列 中 t 出现的个数。 题目数据保证答案符合 32 位带符号整数范围。 示例 1: 输入:s “rabbbit”, t “rabbit” 输出:3 解释&…

智能变电站远程监控解决方案

智能变电站远程监控解决方案 一、项目背景 变电站是改变电压、控制和分配电能的场所,为了加强对变电站的监管力度,抓好供电作业的动态管理,及时、实时的掌握变电站的安全隐患的整改消除情况,确保安全生产贯穿于供电的全过程&…

如何禁用烦人的“insert”键

在我们日常的电脑使用中,有些按键可能会让我们感到非常烦恼,其中最常见的就是“insert”键。这个键位于键盘的右上角,通常会在我们不经意间被按下,导致我们的输入出现了一些奇怪的问题。如果你也曾经遇到过这个问题,那…

在网络安全领域中,主要有哪些技术方向?

入门Web安全、安卓安全、二进制安全、工控安全还是智能硬件安全等等,每个不同的领域要掌握的技能也不同。 当然入门Web安全相对难度较低,也是很多人的首选。主要还是看自己的兴趣方向吧。 本文就以下几个问题来说明网络安全大致学习过程👇 网…

Vue3项目初始配置(更新中ing)

文章目录 别名路径联想配置方法 element plus按需引入并使用风格定制 eslint文件名不强制检测编辑.eslintrc.cjs 别名路径联想 在编写代码的过程中,一旦 输入 / , VSCode会立刻 联想出src下的所有子目录和文件, 统一文件路径访问不容易出 配置方法 在项目的根目…

相亲交友App开发解决方案及功能框架

相亲APP开发功能 1、注册登录:相亲APP开发平台是一款真实的交友软件,所以需要用户提交身份信息进行注册认证,也是为了保障用户的安全! 2、搜索功能:在线、新注册、距离近、同城老乡、星族速配等,都可以在相亲APP内进行…

详解GitHub

详解GitHub 什么是GitHub?GitHub能干什么?1.托管代码、历史版本管理2.搜索开源项目3.分享的同时会得到社区的回馈4.使用Github Pages服务,你可以免费搭建一个博客网站5.学习,能力提升6.提升自己的影响力 GitHub和Git的关系与区别敲…

MySQL基础(三十八)数据库备份与恢复

1 物理备份与逻辑备份 物理备份:备份数据文件,转储数据库物理文件到某一目录。物理备份恢复速度比较快,但占用空间比较大,MySQL中可以用 xtrabackup 工具来进行物理备份。 逻辑备份:对数据库对象利用工具进行导出工作…

Android如何自定义输入文本对话框?

文章目录 0.引言1.创建示例工程2.输入文本对话框布局和功能设计3.主程序调用输入文本对话框 0.引言 笔者研究的课题涉及到安卓软件开发,在开发过程中,发现普通的显示消息对话框一般可以调用android自带包实现,而要通过文本框输入交互&#xf…

代码随想录算法训练营第十一天|20. 有效的括号、1047. 删除字符串中的所有相邻重复项、150. 逆波兰表达式求值

今天的题目都是对栈的经典应用。 有效的括号 题目链接:力扣 解题思路:基于栈结构的特殊性,其非常适合做对称匹配类问题,其实如果知道了要用栈解这道题,在脑中模拟一遍,接下来的思路就是自然而然能够想到…

STM32 多路ADC同时扫描采样

背景 在项目实际应用中,刚好有需求需要使用多路ADC同时采样,这里就选择STM32 ADC多路ADC同时采样,这里简单说明下配置过程,以及使用步骤 原理图 如下图所示,使用四路ADC输入 ADC_Voltage -> 电压信号的采样&…

如何查看linux分区挂载在哪个目录?

一,简介 在linux系统中,如何查看磁盘分区是挂载在哪个目录呢?今天介绍一种方法,供参考。 二,图形化界面查看分区挂载方法 2.1 打开disk工具 2.2 点击查看对应的分区 看一个分区: 如上图所示&#xff0…

百度地图网页设计

一、百度地图api 1、百度搜索——百度地图API 进入——开放平台——开发文档——JavaScript API JavaScript API 首先是GL版本,是三维的效果,我们一般使用二维,选择下面v3.0版本 2、开发指南——注册账号 跟着提示来申请密钥就可。 二、…

Linux基本指令【下】

目录 一、时间相关指令 date显示 时间戳 二、cal指令 三、find指令 (重要)-name 四、grep指令 五、zip\unzip指令 六、tar指令:打包/解压,不打开它,直接看内容 七、bc指令 八、uname -r指令 九、几个重要…

ConstrainLayout(约束布局)属性详解

layout_constraintLeft_toLeftOf layout_constraintRight_toRightOf 这两个基本上用不上因为 layout_constraintStart_toStartOf就相当于layout_constraintLeft_toLeftOf layout_constraintEnd_toEndOf就相当于layout_constraintRight_toRightOf app:layout_constraintBottom_t…

【VMware】Ubantu 22.04配置静态IP

文章目录 一、VMware 虚拟网络配置VMnet8 网络设置注意 关于取消勾选使用本地DHCP服务将IP地址分配给虚拟机VMnet8 NAT设置 网关IP 二、虚拟机 网络适配器三、启动虚拟机 配置网络查看网卡名设置静态IP Reference 一、VMware 虚拟网络配置 VMnet8 网络设置 子网IP子网掩码 编…

22 外部排序

外部排序 外部排序的基本内容 前面介绍过的排序方法都是在内存中进行的(称为内部排序)。外部排序是一种无法全部装入内存的大规模数据集的排序算法。与在内存中处理数据的内部排序相比,外部排序要复杂的多,主要因为是其需要解决…

【c语言】二进制文件的读写操作

创作不易&#xff0c;本篇文章如果帮助到了你&#xff0c;还请点赞 关注支持一下♡>&#x16966;<)!! 主页专栏有更多知识&#xff0c;如有疑问欢迎大家指正讨论&#xff0c;共同进步&#xff01; &#x1f525;c语言系列专栏&#xff1a;c语言之路重点知识整合 &#x…

1.1 IAR新建工作空间 及 新建工程

目录 新建工作空间 新建工程 新建工作空间 &#xff08;1&#xff09;创建一个名字为Workspace的文件夹&#xff0c;如图所示。 &#xff08;2&#xff09;运行IAR EW for 8051 10.10.1&#xff0c;如图所示。 &#xff08;3&#xff09;依次选择File和New Workspace&#xf…