【数据结构】队列及其C语言模拟实现

1.队列的定义

队列是一种一端只能插入数据,一段只能删除数据的数据结构,与栈相反队列是一种先进先出的数据结构,其实也有两边都可以插入与删除的队列这种叫做双端队列,但我这里讲的是普通的队列。

其中插入数据的一端为队尾 ,出数据的一端为队头

2.队列的模拟实现

队列这个数据结构因为两端都要使用,所以明显是使用链表来实现更好,因为如果使用数组来实现的话当某一端要插入或者删除元素的话还要把数组遍历一遍,所以我这里就使用单链表来模拟这个

数据结构

下面是这个队列的定义和我们要实现的函数:

cpp 复制代码
#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <assert.h>

typedef int DataType;

typedef struct QueueNode
{
	struct QueueNode* next;
	DataType val;
}QNode;

typedef struct Queue
{
	QNode* phead;
	QNode* ptail;
	int size;
}Queue;


void QueueInit(Queue* pst);
void QueueDestroy(Queue* pst);

void QueuePush(Queue* pst, DataType x);
void QueuePop(Queue* pst);

DataType QueueFront(Queue* pst);
DataType QueueBack(Queue* pst);

bool QueueEmpty(Queue* pst);
int QueueSize(Queue* pst);

这里我们为什么要另外定义一个结构体Queue呢?因为我们这个基于链表来实现的队列,如果不定义为一个结构体我们需要修改指针时还需要传入二级指针,这样会非常麻烦。其中phead用于表示队头,ptail用于表示队尾

这里我们另外在里面创建一个变量size用于记录我们当前的元素个数,在插入或者删除数据时让这个值相应的++或者--,这样我们就不用遍历这个链表统计个数了,将时间复杂度从O(n)优化为了O(1)的时间复杂度


初始化与销毁

cpp 复制代码
void QueueInit(Queue* pst)
{
	assert(pst);
	pst->phead = pst->ptail = NULL;
	pst->size = 0;
}

void QueueDestroy(Queue* pst)
{
	assert(pst);
	QNode* cur = pst->phead;
	while (cur)
	{
		QNode* next = cur->next;
		free(cur);
		cur = next;
	}
	pst->phead = pst->ptail = NULL;
	pst->size = 0;
}

这里也没有什么好说的,在销毁时记得保留原来要销毁的那个节点指向的下一个地址即可,免得招不到发生内存泄漏的问题

插入与删除

cpp 复制代码
void QueuePush(Queue* pst, DataType x)
{
	assert(pst);
	QNode* newnode = (QNode*)malloc(sizeof(QNode));
	if (newnode == NULL)
	{
		perror("malloc fail !");
		return;
	}
	newnode->val = x;
	newnode->next = NULL;
	if (pst->ptail == NULL)
	{
		pst->ptail = pst->phead = newnode;
	}
	else
	{
		pst->ptail->next = newnode;
		pst->ptail = newnode;
	}
	pst->size++;
}

void QueuePop(Queue* pst)
{
	assert(pst);
	assert(pst->phead);
	if (pst->phead == pst->ptail)
	{
		free(pst->phead);
		pst->phead = pst->ptail = NULL;
	}
	else
	{
		QNode* next = pst->phead->next;
		free(pst->phead);
		pst->phead = next;
	}
	pst->size--;
}

在插入时考虑一下空队列的情况即可,不是空队列则尾插到链表上完成入队列的操作

在执行出队操作时则需要考虑单节点和多节点的情况,如果是单节点的话则将头指针和尾指针都置为空,只需要释放一次内存即可,因为free()是释放指针指向的空间,当只有一个节点是head和tail指向的空间为同一块空间所见仅仅需要释放一次

别忘了让size加减


返回头尾节点

cpp 复制代码
DataType QueueFront(Queue* pst)
{
	assert(pst);
	assert(pst->phead);

	return pst->phead->val;
}

DataType QueueBack(Queue* pst)
{
	assert(pst);
	assert(pst->ptail);

	return pst->ptail->val;
}

剩下的接口都很简单,判断一下头尾指针是否为空就可


是否为空和返回元素个数

cpp 复制代码
bool QueueEmpty(Queue* pst)
{
	assert(pst);

	return pst->size == 0;
}

int QueueSize(Queue* pst)
{
	assert(pst);

	return pst->size;
}

这里就体现的我们在Queue里定义size的智慧了,不用再去遍历链表一遍


相关推荐
Luminous.2 分钟前
C语言--day25
c语言·开发语言
坏柠9 分钟前
从一个设备控制面板开始,系统学习 LVGL 界面开发
android·javascript·学习
QT-Neal9 分钟前
C++智能指针使用详解
开发语言·c++
创业之路&下一个五年9 分钟前
委托、事件、发布-订阅模式全梳理(完整总结)
学习·总结
MartinYeung515 分钟前
[论文学习] 全同态加密下的加密文字比较与子字串搜寻演算法延伸研究
学习·区块链·同态加密
Front思19 分钟前
如何学习Shopify前端开发?
前端·学习
luj_176819 分钟前
硝酸核关联假说缺乏实验证据
c语言·开发语言·c++·经验分享·算法
再玩一会儿看代码29 分钟前
Java浅拷贝和深拷贝理解笔记
java·linux·开发语言·笔记·python·学习
fei_sun31 分钟前
【SystemVerilog验证】数据类型(待补充)
数据结构·systemverilog
草莓熊Lotso32 分钟前
【Linux网络】深入理解 HTTP 协议(三):静态资源服务、状态码与重定向实战
linux·运维·服务器·网络·c++·http