数据结构与算法:队列

在上篇文章讲解了栈之后,本篇也对这一章进行收尾,来到队列!

队列

  • 队列的介绍
  • 队列的存储结构
    • 队列顺序存储的不足之处
  • 循环队列的定义
  • 队列的链式存储结构
    • 链队列的构建
    • 链队列的初始化
    • 队尾入队
    • 队头出队
    • 获取队头队尾元素
    • 判断队列是否为空
    • 获取队列元素个数
    • 队列的销毁

队列的介绍

队列(Queue)就像是排队买票的人群。想象一下你去电影院看电影,人们在售票窗口形成一条线(队列)等待购票。队列遵循一个很重要的原则:先来先服务(First In, First Out,简称FIFO)。这意味着最先到达并排队的人将会是第一个买到票并离开队列的人,随后到达的人则依次排在队伍的后面,等待买票。

客服服务应用了一种数据结构来实现刚才提到的先进先出的排队功能,这就是队列

队列是只允许在一端进行插入操作,而在另一端进行删除操作的线性表

队列是一种先进先出的线性表,允许插入的一端成为队尾,允许删除的一端称为队头

在这里插入图片描述

队列的存储结构

线性表有两种存储结构:顺序存储和链式存储,在栈中我们知道,栈存在两种存储结构,队列作为特殊的线性表,也同样存在这两种存储结构

队列顺序存储的不足之处

我们假设一个队列有n个元素,则顺序存储的队列需要建立一个大于n的数组,并把队列所有元素存储在数组的钱n个单元,数组下表为0的一端即为队头

此时入队列操作,其实就是在队尾追加一个元素,并不需要移动任何元素,时间复杂度为O(1).
在这里插入图片描述

与栈不同的是,队列元素的出列在队头,即下表为0的位置,意味着队列中所有元素都得向前移动。此时时间复杂度为O(N);
在这里插入图片描述
如果不去限制队列元素必须存储在数组前n个单元这一条件,出队的性能则会大大增加,即队头不需要一定要在下标为0的位置。
在这里插入图片描述
此时我们则需要设置队头指针为front,rear指针指向队尾元素的下一个位置

假设长度为5的数组,入队四个元素,rear指针指向下标为4的位置
在这里插入图片描述
出队a1,a2,此时front指向下标为2的位置,rear不变
在这里插入图片描述

当front与rear相等时则队列为空

如果我再入队a5,此时front不变,rear移动到数组之外,指向哪里了呢?
在这里插入图片描述

随着队列操作的进行,如果不断地添加和移除元素,队头指针会向数组的末尾移动,这可能会造成队头不在数组的起始位置。

当继续向队列中添加元素而队尾已经达到数组的最末端时,若不采取任何措施,就无法再添加新的元素,即使数组的前部(队头之前的部分)是空闲的。这种情况看起来好像数组已经“溢出”了,但实际上是因为未充分利用数组的空间,称为“假溢出”。

循环队列的定义

所以我们如何解决上述的假溢出呢?

解决假溢出的办法就是后面满了,再从头开始,头尾相接的循环,把队列这种头尾相接的顺序结构称为循环队列

顺着上述例子,当a5入队后,rear可以改为指向下标0,则解决了指针指向不明

在这里插入图片描述
接着入队a6,将它置于下标为0处,rear指向下标为1处

在这里插入图片描述
上述提到,空队列时,front等于rear,若我插入a7,此时front等于rear,如何判断此时的队列是满还是空呢?
在这里插入图片描述
解决办法:
我们让队列判空条件还是front=rear,当队列满时,我们修改条件,使其保留一个元素空间,也就是队列满时,还有一个空闲单元
在这里插入图片描述
rear可能比front大,也可能比front小,相差一个位置即为满
设队列的最大尺寸为QueueSize,则队列满的条件是
(rear+1)%QueueSize==front

这种顺序存储若不是循环队列,算法性能不高,循环队列又面临着数组溢出的问题,我们接下来讲解队列的链式存储结构**

队列的链式存储结构

队列的链式存储结构,就是线性表的单链表,只不过它只能尾进头出

我们在链队列中有两个指针,一个指向头,一个指向尾

链队列的构建

typedef int QDataType;
typedef struct QueueNode
{QDataType val;struct QueueNode* next;
}QNode;
typedef struct Queue 
{QNode* phead;QNode* ptail;int size;
}Queue;

这里的队列是通过链表实现的,链式存储方式的好处在于它可以动态地分配内存,避免了顺序队列中可能发生的假溢出问题,同时也不需要在队列初始化时就确定其最大容量。

  • phead指针指向队列的头部(第一个元素),而ptail指针指向队列的尾部(最后一个元素)。这两个指针是实现队列基本操作(如入队和出队)的关键

  • size成员存储队列中当前的元素数量。这个信息对于快速获取队列的大小,以及确定队列是否为空等操作非常有用。

链队列的初始化

void QueueInit(Queue* pq)
{assert(pq);pq->phead = pq->ptail = NULL;pq->size=0;
}

在封装的Queue结构体背景下,通过QueueInit(Queue* pq)函数以指针形式传递
传递指向Queue结构体的指针允许QueueInit函数直接对实例进行初始化操作,包括设置头尾指针为NULL和队列大小为0。

队尾入队

只有入队时需要创造新节点,这里我们直接在函数里完成新节点的构造,不需要单独的函数

void QueuePush(Queue* pq, QDataType x)
{assert(pq);QNode* newnode = (QNode*)malloc(sizeof(QNode));if (newnode == NULL){perror("malloc fail");return;}newnode->val = x;newnode->next = NULL;if (pq->ptail == NULL){pq->ptail = pq->phead = newnode;}else{pq->ptail->next = newnode;pq->ptail = newnode;}pq->size++;
}
  • 如果队列为空(即pq->ptail为NULL),则新节点既是队列的头节点也是尾节点,因此将pq->phead和pq->ptail都指向新节点。
  • 如果队列不为空,则将当前尾节点的next指针指向新节点,然后更新pq->ptail指向新节点,这样新节点就成为了队列的尾节点。
  • 将队列的size成员加1,表示队列中元素的数量增加

队头出队

void QueuePop(Queue* pq) {assert(pq);                     if (pq->phead == NULL) return;  QNode* temp = pq->phead;        pq->phead = pq->phead->next;  free(temp);                    temp = NULL;if (pq->phead == NULL) {      pq->ptail = NULL;}pq->size--;  
}
  • if (pq->phead == NULL) return;如果队列为空,则没有元素可以弹出
  • 构建temp中间变量来指向要释放的节点,将头结点指向下一个节点
  • 如果弹出之后队列变为空,则尾指针也要更新为 NULL

获取队头队尾元素

QDataType QueueFront(Queue* pq)
{assert(pq);                     // 确保 pq 不是 NULLassert(pq->phead != NULL);return pq->phead->val; }
QDataType QueueBack(Queue* pq) {assert(pq != NULL); // 确保队列指针不为NULLassert(pq->ptail != NULL); // 确保队列不为空,即队尾指针不为NULL// 返回队列尾部元素的值return pq->ptail->val;
}

这两串获取元素的代码变得十分简单了

判断队列是否为空

bool QueueEmpty(Queue* pq)
{assert(pq);return pq->phead == NULL;
}

获取队列元素个数

int QueueSize(Queue* pq)
{assert(pq);return pq->size;
}

队列的销毁

void QueueDestroy(Queue* pq)
{assert(pq);QNode* cur = pq->phead;while (cur){QNode* next = cur->next;free(cur);cur = next;}pq->phead = pq->ptail = NULL;pq->size = 0;
}

在前面的铺垫之后,我们理解这几个函数就十分简单了

本节内容到此结束!后续我们会更新例题来加强对这个地方的理解!!

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

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

相关文章

Spark解决代码变量bug:error: reassingnment to val

在Scala中,val关键字用于声明一个不可变的变量,一旦赋值后就不能再更改。这就是为什么我尝试重新赋值给modelFilePath时会收到“reassignment to val”的错误。 modelFilePath的声明从val更改为var,因为var允许我们重新赋值 样例代码 val …

267.【华为OD机试真题】贪心歌手(贪心策略-JavaPythonC++JS实现)

🚀点击这里可直接跳转到本专栏,可查阅顶置最新的华为OD机试宝典~ 本专栏所有题目均包含优质解题思路,高质量解题代码(Java&Python&C++&JS分别实现),详细代码讲解,助你深入学习,深度掌握! 文章目录 一. 题目-贪心歌手二.解题思路三.题解代码Python题解代码…

SQL常用函数收藏

日期时间相关 【date_format】 DATE_FORMAT() 函数用于以不同的格式显示日期/时间数据。 参考地址:https://www.runoob.com/sql/func-date-format.html -- 函数:date_format select DATE_FORMAT(Now(),"%Y-%m-%d %H:%i:%S") as v_current_da…

【Java前端技术栈】模块化编程

一、基本介绍 1.基本介绍 1 传统非模块化开发有如下的缺点:(1)命名冲突 (2)文件依赖 2 Javascript 代码越来越庞大,Javascript 引入模块化编程,开发者只需要实现核心的业务逻辑,其他都可以加载别人已经写好的模块 3 Javascrip…

torch.utils.data

整体架构 平时使用 pytorch 加载数据时大概是这样的: import numpy as np from torch.utils.data import Dataset, DataLoaderclass ExampleDataset(Dataset):def __init__(self):self.data [1, 2, 3, 4, 5]def __getitem__(self, idx):return self.data[idx]def…

第2.2章 StarRocks表设计——排序键和数据模型

该篇文章介绍StarRocks-2.5.4版本的数据模型相关内容,有误请指出~ 目录 一、数据模型概述 1.1 四种模型 1.2 排序键 1.2.1 概述 1.2.2 分类 1.2.3 注意事项 二、明细模型 2.1 概述 2.2 适用场景 2.3 建表语句及说明 三、聚合模型 3.1 概述 3.2 适用场…

网络入门基础

本专栏内容为:Linux学习专栏,分为系统和网络两部分。 通过本专栏的深入学习,你可以了解并掌握Linux。 💓博主csdn个人主页:小小unicorn ⏩专栏分类:网络 🚚代码仓库:小小unicorn的代…

[AIGC] JVM内存结构中的方法区主要存储哪些信息?

在JVM的内存结构中,方法区(Method Area)被视为JVM的永久代。它主要负责存储已经被虚拟机加载的类信息、常量、静态变量以及编译器编译后的代码等数据。具体可以分为以下几个部分: 类信息 这部分信息包括类数据(如类名…

32FLASH闪存

目录 一.FLASH简介 二.代码实现 (1)读写内部FLASH (2)读取芯片ID 一.FLASH简介 存储器地址要记得累 系统存储器是原厂写入的Bootloader程序(用于串口下载)&#xff0…

04.结构体和结构体数组

1.结构体 struct stu{int id;string name;float grade; };2.结构体数组的排序 sort算法如果是非自定义类型可以缺省,默认升序排序,但自定义类型必须自己重写比较规则,因为系统没法知道你怎么比较 sort(首元素地址,尾元素下一位…

Python 写网络监控

大家好!我是爱摸鱼的小鸿,关注我,收看每期的编程干货。 网络监控是保障网络可靠性的一项重要任务。通过实时监控网络性能,我们可以及时发现异常,迅速采取措施,保障网络畅通无阻。本文将以 Python为工具&…

Debug系列 GroupNorm和BatchNorm出现Nan或inf的情况

Debug系列 GroupNorm和BatchNorm出现Nan或inf的情况 前言这两个函数做了什么可能出现的问题解决方法train和evalbatchsize或channel设置过小可训练参数的问题数值溢出其它的方法 前言 在复现别人论文的实验结果时,按照README乖乖做完之后,却发现损失函数…

Windows / Linux dir 命令

Windows / Linux dir 命令 1. dir2. dir *.* > data.txt3. dir - list directory contentsReferences 1. dir 显示目录的文件和子目录的列表。 Microsoft Windows [版本 10.0.18363.900] (c) 2019 Microsoft Corporation。保留所有权利。C:\Users\cheng>dir驱动器 C 中…

线性代数:向量组的秩

目录 回顾“秩” 及 向量组线性表示 相关特性 向量组的秩 例1 例2 矩阵的“秩” 及 向量组线性表示 相关特性 向量组的秩 例1 例2

@Async引发的spring循环依赖的问题,

今天发现一个很有意思的问题,正常解决项目中产生的循环依赖,是找出今天添加的注入代码,然后一个个加lazy试过去,会涉及到类中新增的注入 但是今天修改了某个serviceimpl的方法,加入了Async方法后 就发生循环依赖了 ai…

【职场经验】关于自动化用例设计的思考

为什么要设计用例? 作为质量保证(QA)人员,设计测试用例的重要性不亚于开发人员编写技术实现方案。如果实现方案设计不周,编码阶段将可能遇到许多问题;同理,如果我们未能设计测试用例,产品质量就难以得到充…

如何实现一个K8S DevicePlugin?

什么是device plugin k8s允许限制容器对资源的使用,比如CPU和内存,并以此作为调度的依据。 当其他非官方支持的设备类型需要参与到k8s的工作流程中时,就需要实现一个device plugin。 Kubernetes提供了一个设备插件框架,你可以用…

机器视觉系统选型-为什么还要选用工业光源控制器

工业光源控制器最主要的用途是给光源供电,实现光源的正常工作。 1.开关电源启动时,电压是具有波浪的不稳定电压,其瞬间峰值电压超过了LED灯的耐压值,灯珠在多次高压冲击下严重降低了使用寿命; 2.使用专用的光源控制器&…

【算法学习】搜索算法之深度优先搜索

深度优先搜索 DFS 1.算法介绍 深度优先搜索(DFS)算法是一种用于遍历或搜索树或图的算法。它的基本思想是尽可能深地搜索图的分支,直到到达叶节点或无法再深入为止,然后回溯到前一个节点,继续探索其他分支。这种搜索策略可以确保图中的每个节点都被访问到,除非它是一个环。…

inBuilder低代码平台新特性推荐-第十六期

各位友友们,大家好~今天来给大家介绍一下inBuilder低代码平台社区版中的系列特性之一 —— 构件热加载! 01 概述 构件热加载指的是:构件代码修改后,无需重启应用,通过WebIDE的部署或发布工程后,即可正常调…