数据结构与算法:队列

news2025/1/6 13:35:45

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

队列

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

队列的介绍

队列(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 不是 NULL
	assert(pq->phead != NULL);
	return pq->phead->val; 

}
QDataType QueueBack(Queue* pq) {
	assert(pq != NULL); // 确保队列指针不为NULL
	assert(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.coloradmin.cn/o/1461459.html

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

相关文章

2023年12月 Python(六级)真题解析#中国电子学会#全国青少年软件编程等级考试

Python等级考试(1~6级)全部真题・点这里 一、单选题(共25题,共50分) 第1题 运行以下程序,输出的结果是?( ) class A():def __init__(self,x):self.x=x

【MySQL】报错 Incorrect string value: ‘\xE5\xA4\xA9\xE5\x96\x9C‘ for column的解决方法

目录 解决方法如下:例如: 在向数据库中插入中文时遇到报错Incorrect string value: \xE5\xA4\xA9\xE5\x96\x9C for column ,此时为数据库的编码格式有问题,可以按照如下方法修改 解决方法如下: 使用show create table…

【Pytorch深度学习开发实践学习】B站刘二大人课程笔记整理lecture04反向传播

lecture04反向传播 课程网址 Pytorch深度学习实践 部分课件内容: import torchx_data [1.0,2.0,3.0] y_data [2.0,4.0,6.0] w torch.tensor([1.0]) w.requires_grad Truedef forward(x):return x*wdef loss(x,y):y_pred forward(x)return (y_pred-y)**2…

【Web前端笔记10】CSS3新特性

10 CSS3新特性 1、圆角 2、阴影 (1)盒阴影 3、背景渐变 (1)线性渐变(主要掌握这种就可) (2)径向渐变 &…

什么是CODESYS开发系统

CODESYS是一种用于工业自动化领域的开发系统软件,提供了一个完整集成的开发环境。该软件由德国CODESYS GmbH(原 3S-Smart Software Solutions GmbH)公司开发,其最新版本为CODESYS V3。 CODESYS开发系统具有多种特性和优点。首先&a…

【杭州游戏业:创业热土,政策先行】

在前面的文章中,我们探讨了上海、北京、广州、深圳等城市的游戏产业现状。现在,我们切换视角,来看看另一个游戏创业热土——杭州的发展情况 最近第19届亚运会在杭州举办,本次亚运会上,电子竞技首次获准列为正式比赛项…

git push 使用 --mirror 参数复制仓库

迁移一个 Git 仓库并且保留原有的提交记录和分支 克隆原始仓库到本地 git clone <原始仓库URL> <新仓库目录>添加新的远程仓库&#xff1a;git remote add new-origin <新仓库URL>推送所有分支和标签到新的远程仓库&#xff1a;git push new-origin --mirro…

抠人像可抠头发丝的模型-软语义分割(Semantic Human Matting)

软语义分割&#xff08;Semantic Human Matting&#xff09; 一、Semantic Human Matting原理二、Semantic Human Matting 项目文件介绍三、数据集介绍及下载地址四、训练流程五、项目代码下载地址 哔哩哔哩详细解说&#xff08;进主页看全集&#xff09;&#xff1a; https://…

c++入门学习⑦——继承和多态(超级详细版)

目录 前言 继承 继承是什么&#xff1f; 为什么会存在继承&#xff1f; 语法&#xff1a; 一些基本的定义&#xff1a; 三种继承方式&#xff1a; 对象模型 对于构造和析构的顺序 同名函数的处理方式 总结&#xff1a; 静态成员&#xff1a; 定义&#xff1a; 性…

超级详细的python考核试题及答案

一、选择题&#xff08;每题2分&#xff0c;共20分&#xff09; 1&#xff0e;下列哪个语句在Python中是非法的&#xff1f; &#xff08;B&#xff09; A、x y z 1 B、x (y z 1) C、x, y y, x D、x y??xxy 2&#xff0e;关于Python内存管理&#xff0c;下列说法…

创意办公:专注 ONLYOFFICE,探索办公新境界

一.ONLYOFFICE 介绍 ONLYOFFICE 是一个基于 Web 的办公套件&#xff0c;提供了文档处理、电子表格和演示文稿编辑等功能。它被设计为一个协作工具&#xff0c;支持多人实时协作编辑文档&#xff0c;并且可以在本地部署或者作为云服务使用。 二.ONLYOFFICE 特点和功能 以下是 …

机器学习——强化学习作业

作业内容 成功降落在两个黄色旗子中间为成功&#xff0c;其他为失败 Policy Gradient方法 Actor-Critic方法 范例结果 baseline Policy Gradient实现

【Java系列】JDK

目录 JDK介绍JDK版本系列文章版本记录JDK介绍 JDK是 Java 语言的软件开发工具包,主要用于移动设备、嵌入式设备上的java应用程序。JDK是整个java开发的核心,它包含了JAVA的运行环境(JVM+Java系统类库)和JAVA工具。 JDK版本 SE(JavaSE),standard edition,标准版,是我们通…

AOSP10 替换系统launcher

本文实现将原生的launcher 移除&#xff0c;替换成我们自己写的launcher。 分以下几个步骤&#xff1a; 一、新建一个自己的launcher项目。 1.直接使用android studio 新建一个项目。 2.修改AndroidManifest.xml <applicationandroid:persistent"true"androi…

TSINGSEE青犀AI智能分析网关V4初始配置与算法相关配置介绍

TSINGSEE青犀AI智能分析网关V4内置了近40种AI算法模型&#xff0c;支持对接入的视频图像进行人、车、物、行为等实时检测分析&#xff0c;上报识别结果&#xff0c;并能进行语音告警播放。硬件管理平台支持RTSP、GB28181协议、以及厂家私有协议接入&#xff0c;可兼容市面上常见…

Java的String类

目录 String类的常用方法 1.1 字符串构造 1.2 String对象的比较 1.3 字符串查找 1.4 转换 1.5 字符串替换 1.6字符串拆分 1.7 字符串截取 1.8 其他操作方法 1.9 字符串的不可变性 1.10 字符串修改 String类的常用方法 1.1 字符串构造 String类常用的构造方法有很多…

基于springboot+vue的B2B平台的购物推荐网站(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

基于Java (spring-boot)的社区物业管理系统

一、项目介绍 本系统共分为两个角色&#xff1a;管理员和业主。 主要功能有&#xff0c;核心业务处理&#xff0c;基础信息管理&#xff0c;数据统计分析 核心业务处理&#xff1a;车位收费管理&#xff0c;物业收费管理&#xff0c;投诉信息管理&#xff0c;保修信息管理。 …

Vue3之生命周期基础介绍

让我为大家介绍一下vue3的生命周期吧&#xff01; 创建阶段&#xff1a;setup 我们直接console.log就可以了 console.log("创建");挂载阶段&#xff1a;onBeforeMount(挂载前)、onMounted(挂载完毕) import { onBeforeMount, onMounted } from vue; // 挂载前 on…

开源模型应用落地-工具使用篇-向量数据库(三)

一、前言 通过学习"开源模型应用落地"系列文章&#xff0c;我们成功地建立了一个完整可实施的AI交付流程。现在&#xff0c;我们要引入向量数据库&#xff0c;作为我们AI服务的二级缓存。本文将详细介绍如何使用Milvus Lite来为我们的AI服务部署一个前置缓存。 二、术…