当前位置:网站首页>队列的基本操作(C语言实现)
队列的基本操作(C语言实现)
2022-07-01 13:42:00 【Brant_zero】
本篇博客带来栈的好兄弟——队列。
目录
一、 队列的概念与结构
队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出(First In First Out)的特性。入队列:进行插入操作的一端称之为队尾;出队列:进行删除操作的一段称之为队头。

二、 队列的框架定义
队列也可以使用数组或链表实现,使用链表的结构实现更有一些,因为如果使用数组的结构,出队列在数组0下标处上出数据,效率会比较低。
接下来我们来看看此次要实现的队列基本操作。
#pragma once
#include <stdio.h>
#include <stdbool.h>
#include <stdlib.h>
#include <assert.h>
typedef int QDataType;
typedef struct QueueNode
{
struct QueueNode* next;
QDataType data;
}QNode;
typedef struct Queue
{
QNode* head;
QNode* tail;
}Queue;
void QueueInit(Queue* pq); //队列的初始化
void QueueDestroy(Queue* pq); //队列的销毁
void QueuePush(Queue* pq, QDataType x); //插入数据
void QueuePop(Queue* pq); //删除数据
bool QueueEmpty(Queue* pq); //队列的判空
QDataType QueueFront(Queue* pq); //队列头部数据
QDataType QueueBck(Queue* pq); //队列尾部数据
int QueueSize(Queue* pq); //队列的大小注意:
栈使用的是数组,数组是可以直接访问的。而如果我们使用链表实现队列,则要设置一个head和tail指针表示队头和对尾,其中head和tail都为普通的单链表节点结构,所以它们要设置next指针域和data数据域,这样才能实现队列的特性。
三、 队列的功能实现
3.1 队列的初始化
和之前实现的单链表初始化差不多,这里就不过多讲解了。
void QueueInit(Queue* pq)
{
assert(pq);
pq->head = pq->tail = NULL;
}3.2 队列的销毁
因为是链表的形式,所以我们释放空间不能像数组一样直接整段释放,要使用while将每个节点释放掉。
void QueueDestroy(Queue* pq)
{
assert(pq);
QNode* Dest = pq->head;
while (Dest)
{
QNode* temp = Dest->next;
free(Dest);
Dest = temp;
}
pq->head = pq->tail = NULL;
}3.3 队列插入数据
队列插入数据的步骤:
①将newnode链接到head节点的next;②将newnode设置为新的tail节点。
这里我们注意,在我们初始化的时候,我们将head和tail节点都置为了NULL,所以在我们第一次插入数据的时候要做一次判断,如果(head==NULL),则要将tail和head都设置为newnode节点,这样才能保证链表正常链接起来。
注释写的很详细,大家可以看看。
void QueuePush(Queue* pq, QDataType x)
{
assert(pq);
//创建一个节点
QNode* newnode = (QNode*)malloc(sizeof(QNode));
if (newnode == NULL)
{
printf("malloc fail\n");
exit(-1);
}
newnode->data = x;
newnode->next = NULL;
//判断pq是不是一个空结点
//如果是 则将newnode置为当前队列的第一个结点
//如果不是 则将newnode链接到pq的下一个位置
if (pq->tail == NULL)
{ //两个节点都指向newnode 表示此时队列中就一个元素
pq->head = pq->tail = newnode;
}
else
{
//1.将newnode链接到tail的后面
pq->tail->next = newnode;
//2.将newnode置换为新的tail
pq->tail = newnode;
}
}3.4 队列删除数据
上文就提到过,队列是先进先出,所以我们删除数据要从队头删除。
队列删除数据的步骤:
①保存head下一个节点的位置
②释放head节点
③将保存的节点设置为新的head。
注意:
①要先判断队列不能为空
②如果队列中仅剩一个节点的时候,我们按以上的步骤释放了head节点并将其置为了NULL,但是我们没有处理tail节点,此时tail节点就成了经典的野指针。所以我们要额外判断当节点仅剩一个的时候,即(pq->head->next==NULL),我们要释放head结点,并将其head和tail都置为NULL。
代码如下:
void QueuePop(Queue* pq)
{
assert(pq);
assert(!QueueEmpty(pq));
//额外判断一下,解决如果仅剩一个结点了,pop的话导致tail为野指针
if (pq->head->next == NULL)
{
free(pq->head);
pq->head = pq->tail = NULL;
}
else
{
QNode* next = pq->head->next;
free(pq->head);
pq->head = next;
}
}3.5 队列的判空
队列如何判空呢?
此时我们只用判断head是否为NULL即可,即(head->next==NULL);
bool QueueEmpty(Queue* pq)
{
assert(pq);
return pq->head == NULL;
}3.6 取头部数据
注意: 这些取数据和删除数据的操作都要先判断队列是否为空。
QDataType QueueFront(Queue* pq)
{
assert(pq);
assert(!QueueEmpty(pq));
return pq->head->data;
}3.7 取尾部数据
QDataType QueueBck(Queue* pq)
{
assert(pq);
assert(!QueueEmpty(pq));
return pq->tail ->data;
}3.8 队列的大小
链表不数组可以通过计算下标来算长度,所以这里我们只能使用循环计算直到cur==NULL;
int QueueSize(Queue *pq)
{
assert(pq);
int count = 0;
QNode* cur = pq->head;
while (cur)
{
cur = cur->next;
count++;
}
return count;
}四、功能测试
还是和栈一样的测试方式来检查我们队列实现是否成功。
同样,队列是不能遍历的,所以我们也要采用取出一个队列头部数据,然后将头部数据弹出,直到队列为空。

五、总结
相比于栈,队列的性质与栈的截然不同,同样我们这里使用的链式结构与数组的形式也不同。
我感觉实现队列唯一的难点可能就是其中两层结构体的灵活使用吧,其他的与栈的实现很类似,大家可以把队列的框架定义看看,然后自己实现一下,这对于掌握结构体很有帮助,对队列的认识也会加深。
好了,本篇博客到此就结束了,接下来做几道题目之后就会开始树的学习,一段全新的征程,大家一起学习数据结构,一起加油。
边栏推荐
- 微机原理与接口技术知识点整理复习–纯手打
- 10. Page layout, guess you like it
- Error:Kotlin: Module was compiled with an incompatible version of Kotlin. The binary version of its
- When you really learn databinding, you will find "this thing is really fragrant"!
- 2022 · 让我带你Jetpack架构组件从入门到精通 — Lifecycle
- 【NLP】预训练模型——GPT1
- 学历、长相、家境普通的人,未来的发展方向是什么?00后的职业规划都已经整得明明白白......
- 龙蜥社区开源 coolbpf,BPF 程序开发效率提升百倍
- Yarn重启applications记录恢复
- 清华章毓晋老师新书:2D视觉系统和图像技术(文末送5本)
猜你喜欢

Anti fraud, refusing to gamble, safe payment | there are many online investment scams, so it's impossible to make money like this

Content Audit Technology

【NLP】预训练模型——GPT1

Understand the window query function of tdengine in one article

玩转gRPC—不同编程语言间通信

当你真的学会DataBinding后,你会发现“这玩意真香”!

介绍一种对 SAP GUI 里的收藏夹事务码管理工具增强的实现方案

minimum spanning tree

【Flask】Flask启程与实现一个基于Flask的最小应用程序

Fiori 应用通过 Adaptation Project 的增强方式分享
随机推荐
Some summary of pyqt5 learning (overview of the general meaning of some signals and methods)
3.4 data query in introduction to database system - select (single table query, connection query, nested query, set query, multi table query)
Fiori 应用通过 Adaptation Project 的增强方式分享
Social distance (cow infection)
word2vec训练中文词向量
受益互联网出海 汇量科技业绩重回高增长
Research Report on China's software outsourcing industry investment strategy and the 14th five year plan Ⓡ 2022 ~ 2028
清华章毓晋老师新书:2D视觉系统和图像技术(文末送5本)
自定义注解实现验证信息的功能
Listen in the network
Investment analysis and prospect prediction report of global and Chinese p-nitrotoluene industry Ⓙ 2022 ~ 2027
When you really learn databinding, you will find "this thing is really fragrant"!
机器学习总结(一):线性回归、岭回归、Lasso回归
Use of shutter SQLite
Learning to use livedata and ViewModel will make it easier for you to write business
B站被骂上了热搜。。
Analysis report on the development prospect and investment strategic planning of China's wafer manufacturing Ⓔ 2022 ~ 2028
孔松(信通院)-数字化时代云安全能力建设及趋势
20个实用的 TypeScript 单行代码汇总
Analysis report on the development trend and prospect scale of silicon intermediary industry in the world and China Ⓩ 2022 ~ 2027