数据结构之《队列》
在数据结构之《栈》章节中学习了线性表中除了顺序表和链表外的另一种结构——栈,在本篇中我们将继续学习另一种线性表的结构——队列,在通过本篇的学习后,你将会对栈的结构有充足的了解,在了解完结构后我们还将进行栈的实现。一起加油吧!!!

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

那么在队列的底层结构应该选择的是数组还是链表呢?接下来就来分析看看
在使用数组作为队列的底层结构时,由于队列的性质要求先进先出,所以可以使用size来表示数组内的有效元素个数,这样在入队列时就可以直接在数组的size位置插入数据,但是在出队列时为了将数组内的首个元素也就是数组下标为0的位置移除,这就需要将数组从下标为1的位置开始整体都向前移动一位,所以在出队列的时间复杂度为O(N)
在使用单链表作为队列的底层结构时,由于队列的性质要求先进先出,所以如果只用一个phead指针指向链表的第一个节点,那么就会使得在入队列时要找到链表的尾节点就需要通过遍历链表才能实现,这就会使得时间复杂度为O(N),所以为了解决以上这种缺陷,就在创建一个指针指向链表的尾节点,在入队列是就可以直接找到尾节点,这样就可以使得无论是入队列还是出队列时间复杂度都为O(1)
通过以上的分析就可以得出在实现队列时底层结构选择单链表是最优解
2.队列的实现
在实现队列的代码中,在Queue.h头文件中定义队列的结构以及队列中各个函数的声明,在Queue.c文件内完成各个函数的定义,在test.c文件内对实现的各个函数进行测试
2.1队列结构的定义
在队列结构的定义中,创建一个结构体QueueNode来表示节点,该节点内部有两个成员变量,data来存放节点的数据,next指针来存放下一个节点的地址;之后还需再创建一个结构体Queue来存放指向链表的第一个节点和尾节点的指针,并且在这个结构体内还创建一个成员变量size来表示链表中节点的个数,这样是为了在之后的计算队列有效数据个数时直接将size的值提取出来就可实现了
//队列结构的定义
typedef int QDataType;
typedef struct QueueNode
{QDataType data;//节点内的数据struct QueueNode* next;//指向下一个节点的指针}QueueNode;typedef struct Queue
{struct QueueNode* phead;//指向第一个节点的指针struct QueueNode* ptail;//指向尾节点的指针int size;//节点的个数}Queue;
2.2队列的初始化
要完成队列的初始化函数首先要在Queue.h内完成函数的声明
//初始化队列
void QueueInit(Queue* pq);
将该函数命名为QueueInit,函数的参数为存放指向头尾节点指针的结构体指针
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
//初始化队列
void QueueInit(Queue* pq)
{assert(pq);pq->phead = pq->ptail = NULL;pq->size = 0;
}
2.2判断队列是否为空
要完成队列判空函数首先要在Queue.h内完成函数的声明
bool QueueEmpty(Queue* pq);
将该函数命名为QueueEmpty,函数的为存放指向头尾节点指针的结构体指针,函数的放回类型是布尔类型,当队列为空时放回true,不为空就放回false
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
在队列为空时就是指向第一个队列的指针phead和指向队列的尾部的指针ptail都为空,所以该函数的返回值就为pq->phead ==NULL && pq->ptail == NULL
//队列判空
bool QueueEmpty(Queue* pq)
{assert(pq);return pq->phead ==NULL && pq->ptail == NULL;
}
2.3入队列
要完成入队列函数首先要在Queue.h内完成函数的声明
//入队列
void QueuePush(Queue* pq, QDataType x);
将该函数命名为QueuePush,函数的参数有两个,第一个为存放指向头尾节点指针的结构体指针,第二个为要插入的数据
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
在实现数据的入队列前要为数据申请新的节点空间,在此使用malloc来实现,申请完之后在将要入队列的数据x存放到新节点newnode中,之后在将新节点插入到链表时要分以下两种情况,一种是phead指针指向为空时也就是这时链表内一个节点都没有,这时就要将phead和ptail都newnode;另外一种情况是phead指针指向不为空时,这时就先使得ptail的next指针指向newnode,之后再将ptail指向新节点newnode
最后在完成以上操作后要将有效个数size+1
//入队列
void QueuePush(Queue* pq, QDataType x)
{assert(pq);QueueNode* newnode = (QueueNode*)malloc(sizeof(QueueNode));if (newnode == NULL){perror("malloc fail!");exit(1);}newnode->data = x;newnode->next = NULL;if (pq->phead== NULL){pq->phead = pq->ptail = newnode;}else{pq->ptail->next = newnode;pq->ptail = newnode;}pq->size++;
}
2.4出队列
要完成出队列函数首先要在Queue.h内完成函数的声明
//出队列
void QueuePop(Queue* pq);
将该函数命名为QueuePop,函数的参数为存放指向头尾节点指针的结构体指针
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
同时在出队列时队列内不能一个节点都没有,所以要判断队列内不为空,因此要将!QueueEmpty进行assert断言之后在出队列也就是删除单链表的第一个节点时分为以下两种情况,第一种是指针phead和指针ptail指向同一个节点也就是链表中只有一个节点,这时就直接释放该节点,之后再将phead和ptail置为空;另外一种情况是指针phead和指针ptail不相同,这时就先创建一个新的指针变量Next指向原来的phead的next指针指向的节点,之后将phead指向的节点释放后,在将phead指针置为Next指向的节点
最后在完成以上操作后要将有效个数size-1
//出队列
void QueuePop(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));if (pq->phead == pq->ptail){free(pq->phead);pq->phead = pq->ptail = NULL;}else{QueueNode* Next = pq->phead->next;free(pq->phead);pq->phead = Next;}pq->size--;}
2.5取队列头数据
要完成取队列头数据函数首先要在Queue.h内完成函数的声明
//取队列头数据
QDataType QueueFront(Queue* pq);
将该函数命名为QueueFront,函数的参数存放指向头尾节点指针的结构体指针,函数的放回值就为队列的头数据也就是单链表的第一个节点存放的数据
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
同时在出队列时队列内不能一个节点都没有,所以要判断队列内不为空,因此要将!QueueEmpty进行assert断言
在队列中的头数据就为phead指针指向的节点内存放的数据
因此该函数直接放回pq->phead->data即可
//取队列头数据
QDataType QueueFront(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));return pq->phead->data;
}
2.6取队尾数据
要完成取队列尾数据函数首先要在Queue.h内完成函数的声明
//取队列尾数据
QDataType QueueBack(Queue* pq);
将该函数命名为QueueBack,函数的参数为存放指向头尾节点指针的结构体指针,函数的放回值就为队列的尾数据也就是单链表的最后一个节点存放的数据
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
同时在出队列时队列内不能一个节点都没有,所以要判断队列内不为空,因此要将!QueueEmpty进行assert断言
在队列中的尾数据就为ptail指针指向的节点内存放的数据
因此该函数直接放回pq->ptail->data即可
//取队列尾数据
QDataType QueueBack(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));return pq->ptail->data;
}
2.7队列有效数据个数
要完成队列有效数据个数函数首先要在Queue.h内完成函数的声明
//队列有效数据个数
int QueueSize(Queue* pq);
将该函数命名为QueueSize,函数的参数为存放指向头尾节点指针的结构体指针,函数的返回值就为队列内有效的数据个数
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
因为队列内的有效数据个数就为结构体Queue内的成员变量size的值
所以该函数直接放回pq->size即可
//队列有效数据个数
int QueueSize(Queue* pq)
{assert(pq);return pq->size;
}
2.8队列的销毁
要完成队列的销毁函数首先要在Queue.h内完成函数的声明
//销毁队列
void QueueDestory(Queue* pq);
将该函数命名为QueueDestory, 函数的参数为存放指向头尾节点指针的结构体指针
完成了函数的声明接下来就是在Queue.c函数内完成函数的定义
因为以下函数中的pq指针是存放指向头尾节点指针的结构体指针,该指针不能为空,因此要将pq进行assert断言
同时在出队列时队列内不能一个节点都没有,所以要判断队列内不为空,因此要将!QueueEmpty进行assert断言在销毁过程中通过遍历的方法来用free释放链表的所有节点,最后再将指针phead和指针ptail置为空,将有效数据个数赋值为0
//销毁队列
void QueueDestory(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));QueueNode* pcur = pq->phead;while (pcur){QueueNode* Next = pcur->next;free(pcur);pcur = Next;}pq->phead = pq->ptail = NULL;pq->size = 0;
}
3.队列实现完整代码
Queue.h
#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>//队列结构的定义
typedef int QDataType;
typedef struct QueueNode
{QDataType data;//节点内的数据struct QueueNode* next;//指向下一个节点的指针}QueueNode;typedef struct Queue
{struct QueueNode* phead;//指向第一个节点的指针struct QueueNode* ptail;//指向尾节点的指针int size;//节点的个数}Queue;
//初始化队列
void QueueInit(Queue* pq);
//销毁队列
void QueueDestory(Queue* pq);
//入队列
void QueuePush(Queue* pq, QDataType x);
//出队列
void QueuePop(Queue* pq);
//队列判空
bool QueueEmpty(Queue* pq);
//取队列头数据
QDataType QueueFront(Queue* pq);
//取队列尾数据
QDataType QueueBack(Queue* pq);
//队列有效数据个数
int QueueSize(Queue* pq);
Queue.c
#include"Queue.h"//初始化队列
void QueueInit(Queue* pq)
{assert(pq);pq->phead = pq->ptail = NULL;pq->size = 0;
}//销毁队列
void QueueDestory(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));QueueNode* pcur = pq->phead;while (pcur){QueueNode* Next = pcur->next;free(pcur);pcur = Next;}pq->phead = pq->ptail = NULL;pq->size = 0;
}//队列判空
bool QueueEmpty(Queue* pq)
{assert(pq);return pq->phead ==NULL && pq->ptail == NULL;
}//入队列
void QueuePush(Queue* pq, QDataType x)
{assert(pq);QueueNode* newnode = (QueueNode*)malloc(sizeof(QueueNode));if (newnode == NULL){perror("malloc fail!");exit(1);}newnode->data = x;newnode->next = NULL;if (pq->phead== NULL){pq->phead = pq->ptail = newnode;}else{pq->ptail->next = newnode;pq->ptail = newnode;}pq->size++;
}//出队列
void QueuePop(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));if (pq->phead == pq->ptail){free(pq->phead);pq->phead = pq->ptail = NULL;}else{QueueNode* Next = pq->phead->next;free(pq->phead);pq->phead = Next;}pq->size--;}//取队列头数据
QDataType QueueFront(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));return pq->phead->data;
}//取队列尾数据
QDataType QueueBack(Queue* pq)
{assert(pq);assert(!QueueEmpty(pq));return pq->ptail->data;}//队列有效数据个数
int QueueSize(Queue* pq)
{assert(pq);return pq->size;
}相关文章:
数据结构之《队列》
在数据结构之《栈》章节中学习了线性表中除了顺序表和链表外的另一种结构——栈,在本篇中我们将继续学习另一种线性表的结构——队列,在通过本篇的学习后,你将会对栈的结构有充足的了解,在了解完结构后我们还将进行栈的实现。一起…...
【NPU 系列专栏 2 -- NVIDIA 的 H100 和 H200 是什么?】
请阅读【嵌入式及芯片开发学必备专栏】 文章目录 NVIDIA H100 和 H200 芯片NVIDIA H100 芯片简介NVIDIA H100 主要特点NVIDIA H100 应用场景NVIDIA H100 使用举例NVIDIA H200 芯片简介NVIDIA H200 主要特点NVIDIA H200 应用场景NVIDIA H200 使用举例Summary NVIDIA H100 和 H20…...
【BUG】已解决:IndexError: positional indexers are out-of-bounds
IndexError: positional indexers are out-of-bounds 目录 IndexError: positional indexers are out-of-bounds 【常见模块错误】 【解决方案】 原因分析 解决方法 示例代码 欢迎来到英杰社区https://bbs.csdn.net/topics/617804998 欢迎来到我的主页,我是博…...
视频汇聚,GB28181,rtsp,rtmp,sip,webrtc,视频点播等多元异构视频融合,视频通话,视频会议交互方案
现在视频汇聚,视频融合和视频互动,是视频技术的应用方向,目前客户一般有很多视频的业务系统,如已有GB28181的监控(GB现在是国内主流,大量开源接入和商用方案),rtsp设备,音…...
SpringCloud断路器的使用与原理解析
Spring Cloud断路器是在分布式系统中实现容错的一种方式。它的原理是通过在调用链路上添加断路器,当某个服务的调用出现故障或超时时,断路器会自动迅速地切换到快速失败模式,防止故障扩散,从而保护整个系统的稳定性。 Spring Cloud断路器的使用与原理解析如下: 一、使用断…...
结构型模式-分类
一、结构型设计模式 结构型模式描述如何将类或对象按某种布局组成更大的结构。它分为类结构型模式和对象结构型模式,前者采用继承机制来组织接口和类,后者釆用组合或聚合来组合对象。 由于组合关系或聚合关系比继承关系耦合度低,满足“合成…...
【前端】JavaScript入门及实战106-110
文章目录 106 a的索引问题107 使用DOM操作CSS108 读取元素当前的样式109 getStyle()110 其他样式操作的属性滚动条练习 106 a的索引问题 <!DOCTYPE html> <html> <head> <title></title> <meta charset"utf-8"> <script typ…...
git 版本回退-idea
1、选中项目,右键,打开 git历史提交记录 2、选中想要回退的版本,选择 hard(不保留版本记录) 3、最终选择强制提交(必须强制) OK,搞定...
[安洵杯 2019]easy_serialize_php
进入界面然后 <?php$function $_GET[f];function filter($img){$filter_arr array(php,flag,php5,php4,fl1g);$filter /.implode(|,$filter_arr)./i;return preg_replace($filter,,$img); } 这就是个正则if($_SESSION){unset($_SESSION); 销毁 }$_SESSION["use…...
2024年软件测试面试题大全【含答案】
一、面试基础题 简述测试流程: 1、阅读相关技术文档(如产品PRD、UI设计、产品流程图等)。 2、参加需求评审会议。 3、根据最终确定的需求文档编写测试计划。 4、编写测试用例(等价类划分法、边界值分析法等)。 5、用例评审(…...
返回倒数第 k 个节点 - 力扣(LeetCode)
面试题 02.02. 返回倒数第 k 个节点 - 力扣(LeetCode) /*** Definition for singly-linked list.* struct ListNode {* int val;* struct ListNode *next;* };*/int kthToLast(struct ListNode* head, int k) {struct ListNode* fastnode head…...
12 前端工程化
组件化 1. 组件化理解 就是将页面的某一部分独立出来,将这一部分的数据层(M)、视图层(V)和控制层(C)用黑盒的形式全部封装到一个组件内,暴露出一些开箱即用的函数和属性供外部调用。…...
跨文档消息传递:WebKit中的Web通信新纪元
跨文档消息传递:WebKit中的Web通信新纪元 在现代Web应用中,跨文档消息传递(Cross-document messaging)是一种允许不同源的文档进行通信的机制。这种机制对于构建复杂的Web应用,如嵌入式框架(iframes&#…...
面试题 33. 二叉搜索树的后序遍历序列
二叉搜索树的后序遍历序列 题目描述示例 题解递归单调栈 题目描述 输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历结果。如果是则返回 true,否则返回 false。假设输入的数组的任意两个数字都互不相同。 示例 参考以下这颗二叉搜索树&#…...
Web响应式设计———1、Grid布局
1、网格布局 Grid布局 流动网格布局是响应式设计的基础。它通过使用百分比而不是固定像素来定义网格和元素的宽度。这样,页面上的元素可以根据屏幕宽度自动调整大小,适应不同设备和分辨率。 <!DOCTYPE html> <html lang"en"> &l…...
ESP32开发进阶: 训练神经网络
一、网络设定 我们设定一个简单的前馈神经网络,其结构如下: 输入层:节点数:2,接收输入数据,每个输入样本包含2个特征,例如 {1.0, 0.0}, {0.0, 1.0} 等。 隐藏层:节点数:…...
全国区块链职业技能大赛国赛考题前端功能开发
任务3-1:区块链应用前端功能开发 1.请基于前端系统的开发模板,在登录组件login.js、组件管理文件components.js中添加对应的逻辑代码,实现对前端的角色选择功能,并测试功能完整性,示例页面如下: 具体要求如下: (1)有明确的提示,提示用户选择角色; (2)用户可看…...
直接插入排序算法详解
直接插入排序(Straight Insertion Sort)是一种简单直观的排序算法。它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。插入排序在实现上,通常采用in-place排…...
sql手动自增id
有时候在运维处理数据的时候,需要给某张表插入新的记录,那么需要知道最新插入数据的id,并在最新id的基础上加上id增长步长获取新的id,这个过程往往需要现将max出来加1,再手动补充到sql语句中,很麻烦,而且数据多的时候容易出错。有…...
10_TypeScript中的泛型
TypeScript中的泛型) 一、泛型的定义二、泛型函数三、泛型类:比如有个最小堆算法,需要同时支持返回数字和字符串两种类型。通过类的泛型来实现四、泛型接口五、泛型类 --扩展 把类作为参数类型的泛型类1、实现:定义一个 User 的类…...
mongodb源码分析session执行handleRequest命令find过程
mongo/transport/service_state_machine.cpp已经分析startSession创建ASIOSession过程,并且验证connection是否超过限制ASIOSession和connection是循环接受客户端命令,把数据流转换成Message,状态转变流程是:State::Created 》 St…...
【论文笔记】若干矿井粉尘检测算法概述
总的来说,传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度,通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...
EtherNet/IP转DeviceNet协议网关详解
一,设备主要功能 疆鸿智能JH-DVN-EIP本产品是自主研发的一款EtherNet/IP从站功能的通讯网关。该产品主要功能是连接DeviceNet总线和EtherNet/IP网络,本网关连接到EtherNet/IP总线中做为从站使用,连接到DeviceNet总线中做为从站使用。 在自动…...
OPenCV CUDA模块图像处理-----对图像执行 均值漂移滤波(Mean Shift Filtering)函数meanShiftFiltering()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 在 GPU 上对图像执行 均值漂移滤波(Mean Shift Filtering),用于图像分割或平滑处理。 该函数将输入图像中的…...
MySQL账号权限管理指南:安全创建账户与精细授权技巧
在MySQL数据库管理中,合理创建用户账号并分配精确权限是保障数据安全的核心环节。直接使用root账号进行所有操作不仅危险且难以审计操作行为。今天我们来全面解析MySQL账号创建与权限分配的专业方法。 一、为何需要创建独立账号? 最小权限原则…...
2025季度云服务器排行榜
在全球云服务器市场,各厂商的排名和地位并非一成不变,而是由其独特的优势、战略布局和市场适应性共同决定的。以下是根据2025年市场趋势,对主要云服务器厂商在排行榜中占据重要位置的原因和优势进行深度分析: 一、全球“三巨头”…...
初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...
【Android】Android 开发 ADB 常用指令
查看当前连接的设备 adb devices 连接设备 adb connect 设备IP 断开已连接的设备 adb disconnect 设备IP 安装应用 adb install 安装包的路径 卸载应用 adb uninstall 应用包名 查看已安装的应用包名 adb shell pm list packages 查看已安装的第三方应用包名 adb shell pm list…...
鸿蒙(HarmonyOS5)实现跳一跳小游戏
下面我将介绍如何使用鸿蒙的ArkUI框架,实现一个简单的跳一跳小游戏。 1. 项目结构 src/main/ets/ ├── MainAbility │ ├── pages │ │ ├── Index.ets // 主页面 │ │ └── GamePage.ets // 游戏页面 │ └── model │ …...
【Post-process】【VBA】ETABS VBA FrameObj.GetNameList and write to EXCEL
ETABS API实战:导出框架元素数据到Excel 在结构工程师的日常工作中,经常需要从ETABS模型中提取框架元素信息进行后续分析。手动复制粘贴不仅耗时,还容易出错。今天我们来用简单的VBA代码实现自动化导出。 🎯 我们要实现什么? 一键点击,就能将ETABS中所有框架元素的基…...

