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

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

🔥@晨非辰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

Flutter 组件 pos 适配鸿蒙 HarmonyOS 实战:ESC/POS 硬件协议通信,构建高性能零售收银与物联网打印架构

Flutter 组件 pos 适配鸿蒙 HarmonyOS 实战:ESC/POS 硬件协议通信,构建高性能零售收银与物联网打印架构

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 pos 适配鸿蒙 HarmonyOS 实战:ESC/POS 硬件协议通信,构建高性能零售收银与物联网打印架构 前言 在鸿蒙(OpenHarmony)生态迈向专业收银终端、涉及智慧零售设备、数字化仓储标签打印及餐饮自助化服务的背景下,如何实现与热敏打印机(Thermal Printer)等硬件设备的底噪、高可靠通讯,已成为决定商业应用“交易闭环效率”的关键。在鸿蒙设备这类强调硬件直控能力(Raw IO)与实时系统响应的环境下,如果应用依然依赖图形化的截屏打印或低效的中间件转发,由于由于图像编解码的巨大算力消耗,极易由于由于“内存瞬时溢出”导致收银终端在高峰期发生严重卡顿。 我们需要一种能够直接生成热敏指令流(Hex Commands)、支持多种传输通道(蓝牙/Wi-Fi/USB)且符合行业标准(ESC/POS)的高性能控制方案。 pos

By Ne0inhk
手搓简易 Linux 进程池:从 0 到 1 实现基于管道的任务分发系统

手搓简易 Linux 进程池:从 0 到 1 实现基于管道的任务分发系统

🔥草莓熊Lotso:个人主页 ❄️个人专栏: 《C++知识分享》《Linux 入门到实践:零基础也能懂》 ✨生活是默默的坚持,毅力是永久的享受! 🎬 博主简介: 文章目录 * 前言: * 一. 核心设计思路 * 二. 代码模块拆解 * 2.1 任务定义与随机任务生成 * 2.2 子进程任务处理逻辑 * 2.3 通道(Channel)类:封装父子进程通信 * 2.4 进程池(ProcesspPool)类:核心管理逻辑 * 2.5 主函数:进程池使用示例 * 三. 关键知识点解析 * 3.1 管道通信原理 * 3.2 轮询负载均衡 * 3.3 进程回收的坑

By Ne0inhk
Flutter 三方库 lazy_evaluation 的鸿蒙化适配指南 - 深度调优计算性能、实现“按需而动”的极致资源管理方案

Flutter 三方库 lazy_evaluation 的鸿蒙化适配指南 - 深度调优计算性能、实现“按需而动”的极致资源管理方案

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 lazy_evaluation 的鸿蒙化适配指南 - 深度调优计算性能、实现“按需而动”的极致资源管理方案 前言 在高性能应用的开发中,我们常说“最好的优化就是不做无用功”。然而,在复杂的逻辑链中,我们往往会预先计算一堆可能根本不会被用到的变量或模型,这在资源受限的移动设备(尤其是需要极速响应的鸿蒙设备)上是对电池和 CPU 的极大浪费。 惰性求值(Lazy Evaluation)是一种优雅的策略:它确保一个昂贵的计算过程只在程序真正需要其结果时才执行,且结果会被缓存以备后用。 lazy_evaluation 为 Dart 提供了一种极简的封装,完美补齐了编译器层面某些惰性特性的缺失。在 OpenHarmony 系统的适配实操中,我们将看到它如何帮助我们实现更精细的初始化策略,以及如何在确保“鸿蒙式流畅”的同时,极限压榨硬件能效。 一、原理解析 / 概念介绍

By Ne0inhk