重生之我在学数据结构——队列

一.队列的概念

队列是一种先进先出(First In First Out ,FIFO)的数据结构,可以简单理解为排队的概念。在队列中,数据项按照插入的顺序排列,并且只能在队列的一端插入(称为队尾),在另一端删除(称为队头)。

先进先出的概念就是先进队列的数据就会先出队列。比如入队顺序为1、2、3、4,那么出队列的顺序只有1、2、3、4这一种可能性。 

如果我们要选择结构来实现队列的话,应该选谁呢?

数组的话,因为涉及到队头出数据。数组的话,删除第一个数据的话,后面所有的数据都得往前挪,所以我们不考虑数据。

单链表的话,入数据就是尾插,出数据就是头删。队列会频繁的插入与删除数据,而链表相比于数组的有点就在于可以频繁的插入删除。所以我们可以利用单链表来实现队列。

二.队列结构的设计

队列既然要由链表实现,那么我们就要设计节点。节点只需要有数据域(保存队列待存的数据)和指针域(指向下一个节点的指针)。

在队列中,涉及的入数据和出数据分别对应着链表的头和尾,所以我们需要定义一个头节点和一个尾节点。

根据以上两点,我们可以先设计出队列节点的结构:

typedef int QueueDataType;//队列中的节点
typedef struct QueueListNode
{QueueDataType a;struct QueueListNode* next;
}QueueNode;

而对于一条队列来说,我们需要头节点和尾节点,所以我们可以在创建一个结构体Queue,专门用来维护队列。

队列中有的接口是返回有效元素的个数,我们可以在维护队列的结构体中加一个size变量,用来记录当前队列的有效元素个数。如果不这样的话,我们就得遍历链表,会浪费时间。

//队列
typedef struct Queue
{QueueNode* _front;QueueNode* _back;int size;
}Queue;

三.队列的初始化和销毁

3.1队列的初始化

队列初始化,也就是对头节点和尾节点的初始化,我们开始时没有必要开辟空间,只需要在插入数据时在malloc空间即可。所以我们刚开始直接让头尾节点的指针指向NULL,size置为0。

//队列的初始化
void QueueInit(Queue* pq)
{assert(pq);pq->_front = NULL;pq->_back = NULL;pq->size = 0;
}

3.2队列的销毁

我们前面设计,由一条单链表来维护队列,所以队列的销毁其实就是链表的销毁。我们只需要遍历链表,一个节点一个节点的销毁即可。

//队列的销毁
void QueueDestroy(Queue* pq)
{assert(pq);QueueNode* cur = pq->_front;while (cur){QueueNode* next = cur->next;free(cur);cur = next;}pq->_front = NULL;pq->_back = NULL;pq->size = 0;
}

四.入队列

队列插入数据的时候只能从队尾进行插入,对于单链表来说其实就是尾插。要注意的就是,每插入一个数据,size就要++。

//入队尾
void QueuePush(Queue* pq, QueueDataType x)
{assert(pq);//创建新节点QueueNode* newnode = (QueueNode*)malloc(sizeof(QueueNode));if (newnode == NULL){perror("QueuePush()::malloc()");return;}newnode->a = x;newnode->next = NULL;//空队列if (pq->size == 0){pq->_front = newnode;pq->_back = newnode;}else//非空队列{pq->_back->next = newnode;pq->_back = newnode;}pq->size++;
}

五.出队列

队列出数据只能从队头出数据,这样才能保证先进先出规则。而在这里其实就是头删。删除的前提是队列不为空。

头删的时候,我们只需要保存头节点的下一个节点next,然后free掉头节点,再让头节点指向head。

但是这种方式可能会造成野指针。 

当队列只有一个元素的时候,按照这种方式的确会将该节点删除,头节点置为NULL。但是指向尾节点的指针并没有改变指向,也就是说这时候为尾指针成了野指针。等到我们下一次插入数据的话,对尾指针进行解引用就会出错。

所以我们应该判断一下只有一个元素的时候,只有一个元素的时候,直接free掉头节点,然后头指针和尾指针都置为NULL。

删除之后size需要--。

//出队头
void QueuePop(Queue* pq)
{assert(pq);assert(pq->size > 0);//队列不能为空//队列中只有一个元素if (pq->_front == pq->_back){free(pq->_front);pq->_front = NULL;pq->_back = NULL;}else//队列有多个元素{QueueNode* next = pq->_front->next;free(pq->_front);pq->_front = next;}pq->size--;
}

六.获取队头元素

获取队头元素我们直接返回头指针指向节点的数据即可。但是要保证队列不为空。

//获取队头元素
QueueDataType GetQueueFront(Queue* pq)
{assert(pq);assert(pq->size > 0);return pq->_front->a;
}

七.获取队尾元素

获取队尾元素我们直接返回尾指针指向节点的数据即可。同样要保证队列不为空。

//获取队尾元素
QueueDataType GetQueueBack(Queue* pq)
{assert(pq);assert(pq->size > 0);return pq->_back->a;
}

八.获取队列有效元素个数

返回size即可。

//获取队列元素个数
int GetQueueSize(Queue* pq)
{assert(pq);return pq->size;
}

九.判空

判空只需要判断size是否为0即可。

//判空
bool QueueEmpty(Queue* pq)
{assert(pq);return pq->size == 0;
}

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

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

相关文章

品牌策划新手指南:如何让你的品牌脱颖而出?

品牌策划,简单来说,就是为品牌制定一套全方位的成长和发展计划,就像给品牌设计一条成长路线图,让它能够更好地吸引顾客、建立信任、提升知名度,最终实现销售和市场份额的增长。 品牌策划是什么? 想象一下…

【性能优化】在大批量数据下使用 HTML+CSS实现走马灯,防止页面卡顿(一)

切换效果 页面结构变化 1.需求背景 项目首页存有一个小的轮播模块,保密原因大概只能这么展示,左侧图片右侧文字,后端一次性返回几百条数据(开发环境下,生产环境只会更多).无法使用分页解决,前端需要懒加载防止页面卡顿 写个小demo演示,如下 2.解决思路 获取到数据后,取第一…

二、链表(2)

24. 两两交换链表中的节点 法一:迭代,while循环,注意要获取next给变量,得先判断非null, 需要4个变量, n0是前,n1 n2是交换的两,n3是n2的下一个可能为空,这种先把变量保存起来&#…

Leetcode1688. 比赛中的配对次数

问题描述: 给你一个整数 n ,表示比赛中的队伍数。比赛遵循一种独特的赛制: 如果当前队伍数是 偶数 ,那么每支队伍都会与另一支队伍配对。总共进行 n / 2 场比赛,且产生 n / 2 支队伍进入下一轮。如果当前队伍数为 奇…

JavaWeb连接(JDBC)数据库实现增删改查

JavaWeb连接(JDBC)数据库实现增删改查 1、数据库结构 (1)、创建数据库(source_db) (2)、创建数据表(tb_source),结构如下 字段名说明字段类型长度备注id编号int主键,自增,增量为 1name名称v…

前端面试 vue 按钮级的权限控制

方案一 按钮权限也可以用v-if判断 但是如果页面过多,每个页面页面都要获取用户权限role和路由表里的meta.btnPermissions,然后再做判断 这种方式就不展开举例了 方案二 使用自定义指令实现 按钮级的权限控制 思维导图 心就是自定义指令的书写 首先…

【人工智能】Transformers之Pipeline(四):零样本音频分类(zero-shot-audio-classification)

​​​​​​​ 目录 一、引言 二、零样本音频分类(zero-shot-audio-classification) 2.1 概述 2.2 意义 2.3 应用场景 2.4 pipeline参数 2.4.1 pipeline对象实例化参数​​​​​​​ 2.4.2 pipeline对象使用参数 2.4 pipeline实战 2.5 模…

【MySQL】:对库和表的基本操作方法

数据库使用的介绍 什么是SQL 学习数据库的使用——>基于 SQL编程语言 来对数据库进行操作 重点表述的是“需求”,期望得到什么结果。(至于结果是如何得到的,并不关键,都是数据库服务器在背后做好了) 重点表述的是…

线程之间的通信

第一题 #include <stdio.h> #include <string.h> #include <stdlib.h> #include <stdarg.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include <dirent.h> #include <…

蔡司小乐圆:护航青少年视力健康,专业应对近视挑战

在科技日新月异的今天&#xff0c;电子产品已深度融入青少年的日常&#xff0c;为生活带来便利的同时&#xff0c;也悄然间对他们的视力构成了威胁。近视&#xff0c;这一日益严峻的健康问题&#xff0c;正牵动着无数家庭的心弦。蔡司眼镜&#xff0c;作为眼镜行业的领军者&…

7月21日,贪心练习

大家好呀&#xff0c;今天带来一些贪心算法的应用解题、 一&#xff0c;柠檬水找零 . - 力扣&#xff08;LeetCode&#xff09; 解析&#xff1a; 本题的贪心体现在对于20美元的处理上&#xff0c;我们总是优先把功能较少的10元作为找零&#xff0c;这样可以让5元用处更大 …

代码随想录算法训练营第35天|LeetCode 01背包问题 二维、01背包问题 一维、416. 分割等和子集

1. LeetCode 01背包问题 二维 题目链接&#xff1a;https://kamacoder.com/problempage.php?pid1046 文章链接&#xff1a;https://programmercarl.com/背包理论基础01背包-1.html#算法公开课 视频链接&#xff1a;https://www.bilibili.com/video/BV1cg411g7Y6/ 思路&#xf…

压缩pdf大小的方法 指定大小软件且清晰

在数字化时代&#xff0c;pdf文件因其良好的兼容性和稳定性&#xff0c;已成为文档分享的主流格式。然而&#xff0c;高版本的pdf文件往往体积较大&#xff0c;传输和存储都相对困难。本文将为您详细介绍几种简单有效的方法&#xff0c;帮助您减小pdf文件的大小&#xff0c;让您…

UML的六大关系---泛化、实现、关联、聚合、组合、依赖

文章目录 前言1. 泛化关系(Generalization)2. 实现关系(Realization)3. ‌关联关系(Association)4. 聚合关系(Aggregation)5. 组合关系(Composition)6. 依赖关系(Dependency)总结 前言 讲到设计模式&#xff0c;就会有 U M L UML UML类图这个东西。 一开始就很难理解各种线啥意…

Android RSA 加解密

文章目录 一、RSA简介二、RSA 原理介绍三、RSA 秘钥对生成1. 密钥对生成2. 获取公钥3. 获取私钥 四、PublicKey 和PrivateKey 的保存1. 获取公钥十六进制字符串1. 获取私钥十六进制字符串 五、PublicKey 和 PrivateKey 加载1. 加载公钥2. 加载私钥 六、 RSA加解密1. RSA 支持三…

CTF-Web习题:2019强网杯 UPLOAD

题目链接&#xff1a;2019强网杯 UPLOAD 解题思路 打开靶场如下图所示&#xff0c;是一个注册和登录界面 那就注册登录一下&#xff0c;发现是一个提交头像的页面&#xff1a; 试了一下只有能正确显示的png图片才能提交成功&#xff0c;同时F12拿到cookie&#xff0c;base6…

树形背包问题

一些题目给定了树形结构&#xff0c;在这个树形结构中选取一定数量的点或边&#xff08;也可能是其他属性&#xff09;&#xff0c;使得某种与点权或者边权相关的花费最大或者最小。解决这类问题&#xff0c;一般要考虑使用树上背包。 树上背包&#xff0c;顾名思义&#xff0c…

Linux 基础开发工具 : Vim编辑器

Vim 是 Linux 和其他类 Unix 系统上广泛使用的文本编辑器之一。它基于更早的 vi 编辑器&#xff0c;但添加了许多增强功能和扩展。Vim 是“Vi IMproved”的缩写&#xff0c;意为“改进的 Vi”&#xff0c;我们常使用Vim编辑器编写c/c代码。 ps&#xff1a;该篇介绍均为最基础介…

Blender中保存透明图片

在Blender中保存透明图片&#xff0c;主要是通过在渲染设置中调整背景透明度&#xff0c;并选择合适的文件格式来保存图像。以下是一个详细的步骤指南&#xff1a; 一、设置渲染属性 打开Blender并加载你想要渲染的模型。在右侧的属性编辑器中&#xff0c;找到并点击“渲染属…

解决Visual studio内报错信息:MSB8036:找不到 Windows SDK 版本问题

问题描述&#xff1a; 找不到WindowsSDK版本&#xff0c;请安装所需版本的Windows SDK&#xff0c;或者在项目属性页中通过右键单击解决方案并选择“重定解决方案目标”来更改SDK版本。 首先&#xff0c;如果你尝试了以下两种方法&#xff1a; &#xff08;1&#xff09;重新…