首页 > 其他分享 >初阶数据结构【队列及其接口的实现】

初阶数据结构【队列及其接口的实现】

时间:2025-01-14 23:58:53浏览次数:3  
标签:head 初阶 队列 next Queue tail 数据结构 QueueNode

目录


前言

上一期我们讲到了栈,这期我们继续讲解与其相似的结构——队列


一、队列的概念及结构

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

在这里插入图片描述

二、队列的实现方式

队列也可以数组和链表的结构实现,使用链表的结构实现更优一些;

因为如果使用数组的结构,出队列在数组头上出数据,效率会比较低。

在这里插入图片描述

三、队列的实现

3.1 基本结构

由于这里我们用链表实现队列的结构,所以我们先要创建链表节点的结构体:

typedef struct QueueNode
{
	QDataType _data;
	struct QueueNode* _next;
}QueueNode;

我们再创建队列结构体:

typedef struct Queue
{
	QueueNode* _head;
	QueueNode* _tail;
}Queue;

我们通过这样结构体来改变里面的_head或_tail指针的值就可以避免传二级指针;

3.2 队列基本功能接口

初始化队列

// 初始化队列
void QueueInit(Queue* q)
{
	assert(q);
	q->_head = NULL;
	q->_tail = NULL;
}

销毁队列

// 销毁队列
void QueueDestroy(Queue* q)
{
	assert(q);
	QueueNode* cur = q->_head;
	while (cur)
	{
		QueueNode* next = cur->_next;
		free(cur);
		cur = next;
	}
	q->_head = NULL;
	q->_tail = NULL;
}

3.3 入队列接口

注:由于队列只有这个接口需要创建节点,所以我们不再写创建节点接口;

这个接口我们要考虑队列为空的情况:
在这里插入图片描述

// 队尾入队列
void QueuePush(Queue* q, QDataType x)
{
	assert(q);
	QueueNode* newnode = (QDataType*)malloc(sizeof(QueueNode));
	if (newnode == NULL)
	{
		perror("malloc");
		exit(-1);
	}
	newnode->_data = x;
	newnode->_next = NULL;
	if (q->_head == NULL) //当队列为空时
	{
		q->_head = newnode;
		q->_tail = newnode;
	}
	else
	{
		q->_tail->_next = newnode;
		q->_tail = newnode;
	}
}

3.4 出队列接口

这个接口我们要注意当只有一个节点的时候,删除后防止_tail为野指针,所以也要将其置空;
在这里插入图片描述

// 队头出队列
void QueuePop(Queue* q)
{
	assert(q);
	assert(q->_head->_next != NULL); //防止队列为空
	QueueNode* next = q->_head->_next;
	free(q->_head);
	q->_head = next;

	if (q->_head == NULL)
	{
		q->_tail = NULL;
	}
}

3.5 队列的其它接口

获取队列头部元素

QDataType QueueFront(Queue* q)
{
	assert(q);
	assert(q->_head);
	return q->_head->_data;
}

获取队列队尾元素

QDataType QueueBack(Queue* q)
{
	assert(q);
	assert(q->_tail);
	return q->_tail->_data;
}

检测队列是否为空

// 检测队列是否为空,如果为空返回非零结果1,如果非空返回0 
QDataType QueueEmpty(Queue* q)
{
	assert(q);
	return q->_head == NULL ? 1 : 0;
}

获取队列中有效元素个数

QDataType QueueSize(Queue* q)
{
	assert(q);
	QueueNode* cur = q->_head;
	QDataType size = 0;
	while (cur)
	{
		size++;
		cur = cur->_next;
	}
	return size;
}

3.6 测试

void QueueTest()
{
	Queue q;
	QueueInit(&q);
	QueuePush(&q, 1);
	QueuePush(&q, 2);

	printf("%d ", QueueFront(&q));
	QueuePop(&q);

	QueuePush(&q, 3);
	QueuePush(&q, 4);

	//遍历
	while (!QueueEmpty(&q))
	{
		printf("%d ", QueueFront(&q));
		QueuePop(&q);
	}
	printf("\n");
	QueueDestroy(&q);
}

测试结果:
在这里插入图片描述
与预想结果一致,符合;


总结

这期我们用C语言实现了队列的基本操作,我们发现队列与栈有很多相似的地方,在实际应用中我们灵活运用;下期见;
26考研加油!


标签:head,初阶,队列,next,Queue,tail,数据结构,QueueNode
From: https://blog.csdn.net/2303_78558007/article/details/145139009

相关文章

  • 数据结构——链表(概念,类型,java实现、增删、优缺点)
    文章目录链表链表介绍链表类型1.单向链表2.双向链表3.循环链表链表实现(增删改查)链表节点插入节点删除节点链表的特点与优势......
  • 数据结构-链表 day 2
    数据结构-链表单链表一般在算法里面都是采用的静态链表,动态链表单链表一般就是邻接表,包括存储树与图双链表一般是优化某些问题的一下是动态链表与静态链表之间的区别.内存分配方式•静态链表:•静态链表通常是基于一个固定大小的数组来实现的。链表中的每个结点在数......
  • 缓存提速+队列削峰+分库分表+读写分离
    项目背景由于访问量超出设计预期,12306网站在高峰期出现了一系列问题:页面打开缓慢查询和下单报错后台系统过载用户体验不佳根因分析请求高峰(类似于秒杀)响应迟缓  放票时高并发的下单集中在一起,形成请求高峰(类似于秒杀),请求导致订单/电子客票数据......
  • 数据结构—《二叉树的定义与特性》
    目录1.二叉树的定义2.特殊的二叉树1)满二叉树2)完全二叉树3)二叉排序树。4)平衡二又树。5)正则二文树3.二叉树的性质4.二叉树的存储结构1)顺序存储结构2)链式存储结构1.二叉树的定义二叉树是一种特殊的树形结构,其特点是每个结点至多只有两棵子树(即二叉树中不存在度大......
  • 数据结构------树
    前言:前面我们学习了栈和队列。今天我们来学习一种新的数据结构---------树。首先我们来了解一下树的概念。1.树的概念与结构前面我们学习过的顺序表,栈都是一种顺序结构。链表,队列是链式结构。今天学习的树也是一种链式结构。它是由n(n>=0)个有限节点组成一个具有层次关系......
  • 数据结构之链式二叉树
    前言:前面我们学习了树相关的概念及堆的实现,今天来看看二叉树的实现。正式实现二叉树前,我们先了解一些基础知识。对于二叉树基础知识不清楚的可以观看数据结构------树这篇文章,回顾一下知识,有助于理解二叉树。二叉树的遍历方式都有哪些呢?.前序遍历:按照根节点,左节点,右节......
  • 【轻松掌握数据结构与算法】哈希(Hashing)
    什么是哈希?哈希是一种将任意长度的数据转换为固定长度的数据的技术。这个固定长度的数据通常被称为哈希值或哈希码。哈希函数是实现这一转换的关键,它接受任意长度的输入,并产生一个固定长度的输出。为什么使用哈希?哈希的主要用途之一是快速查找数据。通过哈希函数,我们可以将......
  • RabbitMQ-死信队列
    死信,就是无法被消费的消息,一般来说生产者将消息投递到broker或者直接到队列里了,消费者从队列取出消息进行消费。但某些时候由于特定的原因导致队列中的某些消息无法被消费,这样的消息如果没有后续的处理,就变成了死信,有死信自然就有死信队列。死信队列还是队列---只是用来接受特......
  • RabbitMQ-优先级队列及消息配置
    优先级队列C#数据类型queue----先进先出RabbitMQ---队列-----默认也是先进先出~~RabbitMQ设置优先级----可以配置让消费顺序,不按照先进先出的默认规则;给定的优先级---最终体现在消费者;优先级越高,消费的时候,就优先消费。就在前面消费案例:设置{"vip1","hello2","wor......
  • C语言初阶习题(2分支语句和循环语句-for)【10】杨辉三角
    1.题目描述——在屏幕上打印杨辉三角。2.思路第一步先尝试打印下三角第二步,分析他们之间的关系3.代码实现#define_CRT_SECURE_NO_WARNINGS1#include<stdio.h>intmain(){ intn=0; scanf("%d",&n); intarr[100][100]={0}; inti=0; in......