数据结构1-4 队列

一、队列是什么?

先举一个日常例子,排队买饭。

排队买饭

大家按先来后到的顺序,在窗口前排队买饭,先到先得,买完之后走开,轮到下一位买,新来的人排在队尾,不能插队。

可见,上面的“队”的特点是只允许从一端进入,从另一端离开。

这样的一个队,放在数据结构中就是“队列”。

首先,队列是一个线性表,所以它具有线性表的基本特点。

其次,队列是一个受限的线性表,受限之处为:只允许从一端进入队列,从另一端离开。

由此可得: 

        队列Queue,是一种操作受限的线性表,只允许在表的一端进行插入,而在表的另一端进行删除(只允许在队尾添加元素,在队头删除元素,不支持随机访问),向队列中插入元素称为入队或进队;删除元素称为出队或离队,FIFO

相关名词解释:

  • 入队:进入队列,即向队列中插入元素
  • 出队:离开队列,即从队列中删除元素
  • 队头:允许出队(删除)的一端
  • 队尾:允许入队(插入)的一端
  • 队头元素:队列中最先入栈的元素
  • 队尾元素:队列中最后入栈的元素

二、队列的实现思路

和栈一样,队列也可以有两种实现方式:数组实现的顺序队列和链表实现的链队列。

2.1 队列的链式存储

2.1.1 原理

        队列的链式表示称为链队列,实际是一个同时带有队头指针和队尾指针的单链表。

头指针指向队头结点,尾指针指向队尾结点,即单链表的最后一个结点

以看到,要实现一个链队列,需要以下结构:

1.单链表的基本单元结点 —— QueueNode

  • 存储数据的数据域 —— data
  • 指向下一个结点的指针域 —— next

2.指向链表的头指针 —— head

3.标识队头端的队头指针 —— front

4.标识队尾端的队尾指针 —— rear

其中,头指针 head 和队头指针 front 都指向了单链表的第一个结点,所以这个指针可以合二为一,队头指针即头指针。

如此一来,我们可以借助链表的尾插法实现队列的入队操作,借助链表的头删法实现队列的出队操作。

 可参考动画版:Linked List Queue Visualization

链队列入队出队动画

2.1.2 队列的状态

【空队列】:空队列中没有元素,此时,队头下标和队尾下标均为 0,即front = rear = 0: 

【非空非满队列】:队列不是空队列且有剩余空间:

【满队列】:顺序队列分配的固定空间用尽,没有多余空间,不能再插入元素,此时 front = 0,rear = MAXSIZE:

2.1.3 代码实现

typedef int Elemtype;
typedef struct LinkNode {Elemtype data;struct LinkNode *next;
}LinkNode;//先进先出
typedef struct {LinkNode *front, *rear;
}LinkQueue;
2.1.3.1 初始化队列
//队列的初始化,使用的是带头节点的链表
void init_queue(LinkQueue &Q) {Q.front = Q.rear=(LinkNode*)malloc(sizeof(LinkNode));Q.front->next =NULL;}
2.1.3.2 入队
//入队
void enqueue(LinkQueue &Q, Elemtype m) {LinkNode *pnew = (LinkNode *)malloc(sizeof(LinkNode));pnew->data = m;pnew->next =NULL;//要让next 为nullQ.rear->next = pnew;//尾指针next指向pnew,尾插法Q.rear =pnew;//rear指向新的尾部}
2.1.3.3 出队
bool dequeue(LinkQueue &Q, Elemtype &m) {if (Q.front ==Q.rear) {//队列为空return false;}LinkNode *q=Q.front->next;//拿到第一个节点,存入qQ.front->next = q->next;让节点断链m = q->data;if (Q.rear ==q) {Q.rear = Q.front;//链表只剩一个节点时,被删除后要改变rear}free(q);
return true;}
2.1.3.4 主函数
int main() {LinkQueue Q;init_queue(Q);enqueue(Q, 3);enqueue(Q, 4);//   enqueue(Q, 5);Elemtype elem;bool ret;ret = dequeue(Q,elem);if (ret) {printf("dequeue success ele=%d\n",elem);}else {printf("dequeue failed\n");}ret = dequeue(Q,elem);if (ret) {printf("dequeue success ele=%d\n",elem);}else {printf("dequeue failed\n");}ret = dequeue(Q,elem);if (ret) {printf("dequeue success ele=%d\n",elem);}else {printf("dequeue failed\n");}return 0;
}

三、循环队列

3.1 原理

将这种顺序队列画成一个圆:

循环队列的 rear 和 front 能够在队列中一圈一圈地转,像钟表的时针和分针一样。 

【空队列】:队列中没有元素,空队列的条件  front = rear 

【满队列】:少用一个元素,rear + 1 = front

【归零法】:就像钟表的时针满 12 归零一样,front 和 rear 也应该满某个数后归零,这个数就是 MAXSIZE。

比如 rear = 6 时,如果按平常做法来 ,下一步应该是 rear = 7,但在这里,我们让其归零,所以下一步应该是 rear = 0。

用数学公式来表示上面的归零过程就是:rear % MAXSIZE

所以满队列的判断条件应该为:(rear + 1) % MAXSIZE = front。

3.2 循环队列的数组实现

3.2.1 定义

typedef int ElementType;
typedef struct {ElementType data[MaxSize];int front, rear;//队列头,队列尾
}SqQueue;

3.2.2  初始化循环队列

void init_queue(SqQueue &Q) {Q.front =Q.rear = 0;//初始化循环队列,让头尾都指向零号}

3.3.3 判断空队

bool is_empty(SqQueue Q) {return Q.front==Q.rear;}

3.3.4 入队

//入队
bool enqueue(SqQueue &Q,ElementType m ) {//判断循环队列是否满?if ((Q.rear +1) % MaxSize == Q.front){return  false;}Q.data[Q.rear]=m;Q.rear=(Q.rear + 1)%MaxSize;//rear +1 ,如果大于数组最大下标需要回到开头return  true;
}

3.3.5 出队

bool dequeue(SqQueue &Q, ElementType &m) {if (Q.front == Q.rear) {return  false;}m = Q.data[Q.front];Q.front = (Q.front + 1) %MaxSize;return true;}

3.3.6 主函数

int main() {SqQueue Q;init_queue(Q);bool ret;ret= is_empty(Q);if (ret) {printf("SqQueue is empty\n");}else{printf("SqQueue is not empty\n");}enqueue(Q, 3);enqueue(Q, 4);enqueue(Q, 5);enqueue(Q, 6);ret = enqueue(Q, 7);ret =  enqueue(Q, 8);if (ret) {printf("SqQueue success\n");}else{printf("SqQueue failed\n");}ElementType element;ret  =  dequeue(Q, element);if (ret) {printf("dequeue success\n");}else{printf("dequeue failed\n");}ret =  enqueue(Q, 8);if (ret) {printf("SqQueue success\n");}else{printf("SqQueue failed\n");}return 0;
}

 3.3循环队列的链式存储实现(单向循环链表)

队头指针为front,队尾指针为rear;

队空的判断条件:front== rear

队满的判定条件:front == rear->next

3.3.1 代码实战

typedef int ElemType;
typedef struct LNode {ElemType data;struct LNode* next;
}LNode, *LinkList;
3.3.1.1 初始化
void CircleQueue(LinkList &front,LinkList &rear) {front=(LinkList)malloc(sizeof(LNode));rear = front;rear->next = front;EnQueue(front,rear,3);EnQueue(front,rear,3);DeQueue(front,rear);DeQueue(front,rear);DeQueue(front,rear);}
3.3.1.2 入队
void EnQueue(LinkList &front,LinkList &rear, ElemType x) {LinkList pnew;if (rear->next == front) {pnew = (LinkList)malloc(sizeof(LNode));rear->data = x;pnew->next = rear->next;rear->next = pnew;rear = pnew;}else {rear->data =x;rear = rear->next;}
}
3.3.1.2 出队
void DeQueue(LinkList &front,LinkList &rear) {if (front == rear) {printf("the queue is empty\n");}else {printf("the valude=%d\n",front->data);front = front->next;}}

参考地址:https://www.51cto.com/article/656335.html

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

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

相关文章

(十 四)趣学设计模式 之 策略模式!

目录 一、 啥是策略模式?二、 为什么要用策略模式?三、 策略模式的实现方式四、 策略模式的优缺点五、 策略模式的应用场景六、 总结 🌟我的其他文章也讲解的比较有趣😁,如果喜欢博主的讲解方式,可以多多支…

探秘基带算法:从原理到5G时代的通信变革【三】Turbo 编解码

文章目录 2.2 Turbo 编解码2.2.1 基本概念与系统构成2.2.2 编码过程分步解析交织器递归系统卷积编码器复接器总结 2.2.3 译码算法分类与原理Turbo码的强大主要来源于其解码器理论基础解码过程详解交织与解交织译码算法总结 2.2.4 Turbo码的应用场景无线通信卫星通信深空通信 2.…

Yocto + 树莓派摄像头驱动完整指南

—— 从驱动配置、Yocto 构建,到 OpenCV 实战 在树莓派上运行摄像头,在官方的 Raspberry Pi OS 可能很简单,但在 Yocto 项目中,需要手动配置驱动、设备树、软件依赖 才能确保摄像头正常工作。本篇文章从 BSP 驱动配置、Yocto 关键…

TCP协议(20250304)

1. TCP TCP: 传输控制协议(Transmission Control Protocol),传输层协议之一(TCP,UDP) 2. TCP与UDP UDP(用户数据报协议) 面向数据报无连接不安全不可靠(尽最大努力交…

NModbus 连接到Modbus服务器(Modbus TCP)

1、在项目中通过NuGet添加NModbus,在界面中添加一个Button。 using NModbus.Device; using NModbus; using System.Net.Sockets; using System.Text; using System.Windows; using System.Windows.Controls; using System.Windows.Data; using System.Windows.Docu…

【零基础到精通Java合集】第十八集:多线程与并发编程-线程池与Callable/Future应用

课程标题:线程池与Callable/Future应用(15分钟) 目标:掌握线程池的创建与管理,理解Callable任务与Future异步结果处理机制 0-1分钟:课程引入与线程池意义 以“银行窗口服务”类比线程池:复用固定资源(柜员)处理多任务(客户)。说明线程池的核心价值——避免频繁创建…

【leetcode hot 100 238】除自身以外数组的乘积

解法一:(左右乘积列表)利用索引左侧所有数字的乘积和右侧所有数字的乘积(即前缀与后缀)相乘得到答案。 class Solution {public int[] productExceptSelf(int[] nums) {int len nums.length;int[] L new int[len]; …

BUU44 [BJDCTF2020]ZJCTF,不过如此1 [php://filter][正则表达式get输入数据][捕获组反向引用][php中单双引号]

题目: 我仿佛见到了一位故人。。。也难怪,题目就是ZJCTF 按要求提交/?textdata://,I have a dream&filenext.php后: ......不太行,好像得用filephp://filter/convert.base64-encode/resourcenext.php 耶?那 f…

[Web 安全] PHP 反序列化漏洞 —— POP 链构造思路

关注这个专栏的其他相关笔记:[Web 安全] 反序列化漏洞 - 学习笔记-CSDN博客 0x01:什么是 POP 链? POP 链(Payload On Purpose Chain)是一种利用 PHP 中的魔法方法进行多次跳转以获取敏感数据的技术。它通常出现在 CTF…

扩散语言模型:从图像生成到文本创造的范式跃迁

近年来,扩散模型(Diffusion Models)在人工智能领域异军突起,尤其在图像生成任务中取得了令人瞩目的成就,如 Stable Diffusion 等模型已成为生成高质量图像的标杆。这种成功激发了研究者们的好奇心:扩散模型的魔力能否从视觉领域延伸至自然语言处理(NLP),为文本生成带来…

大模型工程师学习日记(十):基于 LangChain 构建向量存储和查询 Qdrant

Qdrant介绍 Qdrant(读作:quadrant /kwɑdrənt/ n. 象限;象限仪;四分之一圆)是一个向量相似度搜索引擎。它提供了一个生产就绪的服务,具有方便的 API 来存储、搜索和管理点 - 带有附加载荷的向量。Qdrant专…

DeepSeek 助力 Vue3 开发:打造丝滑的网格布局(Grid Layout)

前言:哈喽,大家好,今天给大家分享一篇文章!并提供具体代码帮助大家深入理解,彻底掌握!创作不易,如果能帮助到大家或者给大家一些灵感和启发,欢迎收藏关注哦 💕 目录 Deep…

deepseek、腾讯元宝deepseek R1、百度deepseekR1关系

分析与结论 区别与联系 技术基础与定制方向: DeepSeek官网R1版本:作为基础版本,通常保留通用性设计,适用于广泛的AI应用场景(如自然语言处理、数据分析等)。其优势在于技术原生性和官方直接支持。腾讯元宝…

外贸独立站使用wordpress模板与定制哪个SEO效果好

使用WordPress模板搭建的外贸独立站与定制站的SEO效果,可以从以下几个方面进行分析: 1. 内容质量是SEO的核心 内容质量确实是SEO的关键,无论使用模板还是定制开发,优质、相关、原创的内容都是提升排名的基础。内容能够解决用户问…

Golang语法特性总结

1.认识Golang代码特性 package main //1.包含main函数的文件就是一个main包--当前程序的包名// import "fmt" // import "time" import("fmt""time" )//3.同时包含多个包 4.强制代码风格:函数的 { 一定和函数名在同一行,否…

AI赋能校园安全:科技助力预防与应对校园霸凌

校园本应是学生快乐学习、健康成长的地方,然而,校园霸凌却成为威胁学生身心健康的隐形“毒瘤”。近年来,随着人工智能(AI)技术的快速发展,AI在校园安全领域的应用逐渐成为解决校园霸凌问题的新突破口。通过…

易语言模拟真人鼠标轨迹算法 - 防止游戏检测

一.简介 鼠标轨迹算法是一种模拟人类鼠标操作的程序,它能够模拟出自然而真实的鼠标移动路径。 鼠标轨迹算法的底层实现采用C/C语言,原因在于C/C提供了高性能的执行能力和直接访问操作系统底层资源的能力。 鼠标轨迹算法具有以下优势: 模拟…

运营商三要素API:构建安全信任的桥梁

引言 在数字经济时代,身份验证已成为各类业务场景的基础需求。运营商三要素API作为一种高效的身份核验工具,通过对接运营商数据,实现对用户姓名、身份证号码、手机号码三项关键信息的实时校验,为各行业提供可靠的身份认证解决方案…

Spring Boot 与 MyBatis 版本兼容性

初接触Spring Boot,本次使用Spring Boot版本为3.4.3,mybatis的起步依赖版本为3.0.0,在启动时报错,报错代码如下 org.springframework.beans.factory.BeanDefinitionStoreException: Invalid bean definition with name userMapper…

GCN从理论到实践——基于PyTorch的图卷积网络层实现

Hi,大家好,我是半亩花海。图卷积网络(Graph Convolutional Network, GCN)是一种处理图结构数据的深度学习模型。它通过聚合邻居节点的信息来更新每个节点的特征表示,广泛应用于社交网络分析、推荐系统和生物信息学等领…