【数据结构入坑指南(六)】--《从初始化到销毁:手把手教你打造健壮的队列实现》

【数据结构入坑指南(六)】--《从初始化到销毁:手把手教你打造健壮的队列实现》

🔥@晨非辰Tong:个人主页 

👀专栏:《C语言》《数据结构与算法》《数据结构与算法刷题集》

💪学习阶段:C语言、数据结构与算法初学者

⏳“人理解迭代,神理解递归。”


引言:刚征服了“后进先出”的栈,现在让我们迎接一个全新的挑战——队列。这个“先进先出”的数据结构将带你体验截然不同的设计思维。本文将手把手带你用链表实现一个队列,为后续学习树形结构打下坚实基础。

目录

 一、队列初探:核心概念与结构设计

1.1  深入理解“先进先出”(FIFO)

1.1.1  关键抉择:链表 vs 数组

1.2  搭建队列的“骨架”

二、核心功能实现:从零搭建完整队列

2.1  准备工作:搭建稳固的基础

2.1.1  队列初始化:一切从这里开始

2.1.2  判空检测:守护队列的“安全门”

2.2   入队操作:在队尾优雅地添加数据

2.3  出队操作:从队头安全地移除数据

2.4   销毁队列:善始善终的内存管理

三、功能扩展:让队列更加强大

3.1  窥探队首:获取但不破坏

3.2  窥探队尾:快速访问末端元素

3.3 清点元素:高效统计队列大小

四、完整代码展示:理论与实践的完美结合

4.1  Queue.h:队列的“蓝图”与接口

4.2  Queue.c:核心逻辑的完整实现

4.3  test.c:功能验证与测试用例


 一、队列初探:核心概念与结构设计

1.1  深入理解“先进先出”(FIFO)

--概念:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有“先进先出”与栈截然不同的的特点。

  • 入队列:进行插入操作的一端称为队尾;
  • 出队列:进行删除操作的一端称为队头;

1.1.1  关键抉择:链表 vs 数组

--那么实现队,底层结构选择数组实现还是链表实现呢?

        :那当然是都可以实现,此时—>二者的插入、删除算法操作总会有一个时间复杂度为O(1)。但是链表有补救操作(尾插为O(N))——>定义一个指针始终指向尾节点,这就不会导致还要进行循环遍历找到尾节点来插入。

1.2  搭建队列的“骨架”

--结构,因为底层实现是链表,就要将节点结构、队列结构一同定义。

typedef int QDataType; //链表节点结构 typedef struct QueueNode { QDataType data; struct QueueNode* next; }QueueNode; //队列结构 typedef struct Queue { QueueNode* phead; QueueNode* ptail; }Queue;

二、核心功能实现:从零搭建完整队列

--下面也是实现一些擦插入、删除操作,先进行初始化

2.1  准备工作:搭建稳固的基础

2.1.1  队列初始化:一切从这里开始

--初始,头、尾指针均指向NULL。

//初始化 void QueueInit(Queue* pq) { assert(pq);//防止空指针 pq->phead = pq->ptail = NULL; }

2.1.2  判空检测:守护队列的“安全门”

--此操作,在出队列时,判断是否有节点。

bool QueueEmpty(Queue* pq) { assert(pq); return pq->phead == NULL; }

2.2   入队操作:在队尾优雅地添加数据

Queue.c #include "Queue.h" //入队(尾插) void QueuePush(Queue* pq, QDataType x) { assert(pq); //创建新节点 QueueNode* newnode = (QueueNode*)malloc(sizeof(QueueNode)); if (newnode == NULL) { perror("mallpc 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 = pq->ptail->next;//尾指针移动到新节点 } } test.c #include "Queue.h" void test01() { Queue q; QueueInit(&q); QueuePush(&q, 1); QueuePush(&q, 2); QueuePush(&q, 3); QueuePush(&q, 4); } int main() { test01(); return 0; }
--插入节点就要先申请一份节点大小的空间,其中节点结构:Data存要求插入的数据,next指 NULL。

--刚开始,队列中没有节点,新插入的节点为第一节点,导致头指针、尾指针都指向新节点。

--以后直接是尾指针 next 指向新节点,然后尾指针移动到新节点。

2.3  出队操作:从队头安全地移除数据

Queue.c #include "Queue.h" //出队列 void QueuePop(Queue* 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; } } void test01() { Queue q; QueueInit(&q); QueuePush(&q, 1); QueuePush(&q, 2); QueuePush(&q, 3); QueuePush(&q, 4); QueuePop(&q); QueuePop(&q); QueuePop(&q); QueuePop(&q); } int main() { test01(); return 0; }
--操作前,先判断队列是否有节点,有节点才能出队列。

--没有节点,头指针、尾指针都指向空。

--否则,创建新 next 指针将第二节点的地址先保存(也就是头指针指向的结构体中的 next 指针保存着地址),释放后,头指针指向新指针。

2.4   销毁队列:善始善终的内存管理

--销毁队列,需要遍历队列依次释放空间(注意提前保存下一节点的地址),最后将头指针、尾指针置空。

Queue.c #include "Queue.h" //销毁 void QueueDesTroy(Queue* pq) { assert(pq); QueueNode* pcur = pq->phead; while (pcur) { //保存下一节点的地址 QueueNode* next = pcur->next; free(pcur); pcur = next; } pq->phead = pq->ptail = NULL; }

三、功能扩展:让队列更加强大

3.1  窥探队首:获取但不破坏

--因为事先已经定义了指向头节点的指针,且指向的结构体中存在存储当前节点数据的 data 变量,直接返回 data 即可。

Queue.c #include "Queue.h" //取队头数据 QDataType QueueFront(Queue* pq) { assert(!QueueEmpty(pq)); return pq->phead->data; } test.c #include "Queue.h" void test01() { //初始化 Queue q; QueueInit(&q); //入队 QueuePush(&q, 1); QueuePop(&q); QueuePush(&q, 2); QueuePop(&q); QueuePush(&q, 3); QueuePop(&q); QueuePush(&q, 4); //取队首 printf("%d\n", QueueFront(&q)); } int main() { test01(); return 0; } 

--随着phead指向的改变,其指向的结构体中访问到的的data、next也随之改变。

3.2  窥探队尾:快速访问末端元素

--与上面的取队首数据算法思路一样,也是事先定义了指向尾节点的指针,且指向的结构体中存在存储当前节点数据的 data 变量,直接返回 data 即可。

Queue.c #include "Queue.h" //取队尾数据 QDataType QueueBack(Queue* pq) { assert(!QueueEmpty(pq)); return pq->ptail->data; } test.c #include "Queue.h" void test01() { //初始化 Queue q; QueueInit(&q); //入队 QueuePush(&q, 1); QueuePush(&q, 2); QueuePush(&q, 3); QueuePush(&q, 4); //取队尾 printf("%d->", QueueBack(&q)); } int main() { test01(); return 0; }

3.3 清点元素:高效统计队列大小

--这就需要进行循环内遍历到尾节点,统计个数

Queue.c #include "Queue.h" //队列有效元素个数 int QueueSize(Queue* pq) { assert(pq); QueueNode* pcur = pq->phead; int size = 0; while (pcur) { ++size; pcur = pcur->next; } return size; }


四、完整代码展示:理论与实践的完美结合

4.1  Queue.h:队列的“蓝图”与接口

#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 { QueueNode* phead;//指向队头 QueueNode* ptail;//指向队尾 }Queue; //初始化 void QueueInit(Queue* pq); //判队列是否为空 bool QueueEmpty(Queue* pq); //入队(尾插) void QueuePush(Queue* pq, QDataType x); //出队列(队头) void QueuePop(Queue* pq); //取队头数据 QDataType QueueFront(Queue* pq); //取队尾数据 QDataType QueueBack(Queue* pq); //队列有效元素个数 int QueueSize(Queue* pq); 

4.2  Queue.c:核心逻辑的完整实现

#include "Queue.h" //初始化 void QueueInit(Queue* pq) { assert(pq);//不能传空指针 pq->phead = pq->ptail = NULL; } //入队(尾插) void QueuePush(Queue* pq, QDataType x) { assert(pq); //创建新节点 QueueNode* newnode = (QueueNode*)malloc(sizeof(QueueNode)); if (newnode == NULL) { perror("mallpc 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 = pq->ptail->next;//尾指针移动到新节点 } } //判队列是否为空 bool QueueEmpty(Queue* pq) { assert(pq); return pq->phead == NULL; } //出队列 void QueuePop(Queue* 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; } } //取队头数据 QDataType QueueFront(Queue* pq) { assert(!QueueEmpty(pq)); return pq->phead->data; } //取队尾数据 QDataType QueueBack(Queue* pq) { assert(!QueueEmpty(pq)); return pq->ptail->data; } //队列有效元素个数 int QueueSize(Queue* pq) { assert(pq); QueueNode* pcur = pq->phead; int size = 0; while (pcur) { ++size; pcur = pcur->next; } return size; } //销毁 void QueueDesTroy(Queue* pq) { assert(pq); QueueNode* pcur = pq->phead; while (pcur) { //保存下一节点的地址 QueueNode* next = pcur->next; free(pcur); pcur = next; } pq->phead = pq->ptail = NULL; }

4.3  test.c:功能验证与测试用例

#include "Queue.h" void test01() { //初始化 Queue q; QueueInit(&q); //入队 QueuePush(&q, 1); QueuePush(&q, 2); QueuePush(&q, 3); QueuePush(&q, 4); //出队 /*QueuePop(&q); QueuePop(&q); QueuePop(&q); QueuePop(&q);*/ //取队首 /*printf("%d->", QueueFront(&q)); QueuePop(&q); printf("%d->", QueueFront(&q)); QueuePop(&q); printf("%d->", QueueFront(&q)); QueuePop(&q); printf("%d->", QueueFront(&q));*/ //取队尾 //printf("%d->", QueueBack(&q)); //有效数据 printf("size:%d\n", QueueSize(&q)); } int main() { test01(); return 0; }

回顾:

【面试高频数据结构(五)】--《手把手实现栈结构:附带完整代码与注释,深度揭秘数组实现香在哪?》

【面试高频数据结构(四)】--《超越单链表的局限:双链表“哨兵位”设计模式,如何让边界处理代码既优雅又健壮?》
结语:队列的实现标志着你在线性数据结构领域已登堂入室。接下来,二叉树与堆的学习将带你进入非线性数据结构的新世界。有趣的是,队列正是基于堆实现的——现在打下的队列基础,将在下一章绽放新的光彩。

Read more

C++测试与调试:确保代码质量与稳定性

C++测试与调试:确保代码质量与稳定性

C++测试与调试:确保代码质量与稳定性 一、学习目标与重点 本章将深入探讨C++测试与调试的核心知识,帮助你确保代码的质量与稳定性。通过学习,你将能够: 1. 理解测试与调试的基本概念,掌握测试方法和工具 2. 学会使用单元测试框架,如Google Test和Catch2 3. 理解集成测试的重要性,确保系统的功能正确性 4. 学会使用调试工具,如GDB和Visual Studio调试器 5. 培养测试与调试思维,设计高质量的代码 二、测试的基本概念 2.1 测试的分类 测试可以分为以下几类: * 单元测试:测试单个函数或类的功能 * 集成测试:测试多个模块的集成功能 * 系统测试:测试整个系统的功能 * 验收测试:测试系统是否满足用户需求 * 性能测试:测试系统的性能指标 2.2 测试原则 测试应该遵循以下原则: * 测试应该尽可能早地进行 * 测试应该覆盖所有可能的场景 * 测试应该是自动化的

By Ne0inhk
Redis 核心数据结构:String 类型深度解析与 C++ 实战

Redis 核心数据结构:String 类型深度解析与 C++ 实战

Redis 核心数据结构:String 类型深度解析与 C++ 实战 前言 在当今数据驱动的世界里,Redis 以其卓越的性能和丰富的数据结构,已成为内存数据库领域的翘楚。无论是作为高速缓存、消息队列,还是分布式锁的实现方案,Redis 的身影无处不在。而在 Redis 提供的所有数据结构中,String 类型无疑是基石中的基石。它不仅是构建其他复杂结构的基础,其自身强大的命令集也足以应对各种复杂的业务场景。 本文将以广受欢迎的 C++ Redis 客户端库 redis-plus-plus 为实战工具,系统性地、由浅入深地剖析 Redis String 类型的核心命令。我们将从最基础的 SET 和 GET 操作讲起,逐步探索包括过期时间设置、条件更新、批量操作、子字符串处理以及原子计数器在内的各种高级用法。 本文旨在为您提供一份不仅包含“如何做”,更解释“为什么这么做”的详尽指南。我们将深入探讨 redis-plus-plus

By Ne0inhk
【C++藏宝阁】C++入门:命名空间(namespace)详解

【C++藏宝阁】C++入门:命名空间(namespace)详解

🌈个人主页:聆风吟 🔥系列专栏:C++藏宝阁 🔖少年有梦不应止于心动,更要付诸行动。 文章目录 * 📚专栏订阅推荐 * 📋前言:为什么需要命名空间? * 一、命名空间的定义 * 二、命名空间的使用 * 三、命名空间的特性 * 3.1 命名空间的嵌套定义 * 3.2 命名空间的定义可以不连续 * 四、命名空间的本质:独立的作用域 * 4.1 命名空间是C++的一种作用域类型 * 4.2 命名空间作用域的特点 * 4.3 域作用限定符 `::` 的作用 * 4.4 编译器的查找规则 * 五、命名空间的价值 * 5.1 解决命名冲突 * 5.2 模块化组织代码 * 5.3

By Ne0inhk
Java滑动窗口算法题目练习

Java滑动窗口算法题目练习

滑动窗口算法 * 长度最小的子数组 * 无重复字符的最长子串 * 最大连续1的个数||| * 将x减到0的最小操作数 * 水果成蓝 * 找到字符串中所有字母异位词 * 串联所有单词的子串 * 最小覆盖子串 长度最小的子数组 题目解析:这里给我们一个全是正整数的数组和一个目标值,让我们找一段连续的区间,这个区间的值之和是大于等于目标值的,从这个数组中找到一个最小的区间长度,如果不存在的话就返回0 算法原理:1.首先我们是可以使用暴力解法,双重for循环进行遍历出所有的情况,当满足区间的值大于等于目标值的话就进行结果更新,反之继续向后操作,我们可以发现这里是有许多的是多余的,就像如果此时的这个区间的值已经大于这个目标值了,如果继续向后操作的话,这个数组是正整数数组,肯定还是大于等于目标值,但是长度变长了,我们要找到是最短的,因此我们可以不需要让其重复操作,直接开始下一次循环就行 2."同向双指针"也叫滑动窗口算法,这里我们可以使用left和right两个指针向同一个方向移动,并且不回退,此时的思想就和上面暴力解法优化思想一样,一直进行将right下标对应的值放入

By Ne0inhk