数据结构【队列】

news/2024/5/14 22:36:41/文章来源:https://blog.csdn.net/m0_59292239/article/details/127847566

文章目录

  • (一)队列定义
  • (二)队列实现
    • (1)创建结构体
    • (2)具体函数实现及解析
      • 1.1 初始化队列
      • 1.2入队列
      • 1.3出队列
      • 1.4取队首元素
      • 1.5取队尾元素
      • 1.6返回队列个数
      • 1.7判断是否为空
      • 1.8销毁队列
  • (三)队列实现代代码
    • (1)Queue.c
    • (2)Queue.h
    • (3)test.c
  • (四)队列测试结果

(一)队列定义

队列是一种常用的数据结构,也是一种操作受限制的线性表,特点是只允许在表的头部进行删除操作,在表的尾部进行插入操作,队列具有先进先出FIFO(First In First Out)。

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

我们实现可以用数组和链表结构实现,但使用链表更优,如果去用数组实现啊,出队列在数组头上出数据,效率会很低 ,需要挪动n次,时间复杂度为O(N)。

(二)队列实现

(1)创建结构体

typedef int QDataType;
typedef struct QueueNode
{QDataType data;struct QueueNode* next;
}QNode;
typedef struct Queue
{QNode* head;QNode* tail;int size;
}Queue;

我们要创建两个结构体,一个表示链式结构,即队列,第二个是队列的结构。
先使用typedef int QDateType,是为了方便改类型,在结构体里创建2个成员变量,分别表示结点的数据域和指针域,之后创建队列里的结构体,在里面创建两个指针变量分别指向队列的头部和尾部,size记录队列的个数。

(2)具体函数实现及解析

1.1 初始化队列

void QueueInit(Queue* qq)//队列初始化
{assert(qq);qq->head = qq->tail = NULL;qq->size = 0;}

初始化队列,传一级指针改变的是结构体,只需要结构体指针,qq是不能为空的,所以需要断言防止人为穿空,把头指针和尾指针置空,size置0。

1.2入队列

void QueuePush(Queue* qq, QDataType x)//入队列
{assert(qq);QNode* newnode = (QNode*)malloc(sizeof(QNode));if (newnode == NULL){perror("malloc fail");}newnode->data = x;newnode->next = NULL;if (qq->tail == NULL){qq->head = qq->tail = newnode;}else{qq->tail->next = newnode;qq->tail = qq->tail->next;}qq->size++;
}

入队列其实就是尾插,先malloc申请一个结点,再判断是否为空,再把结点数据域和指针域分别给上x和空,再进行尾插:判断尾或者头是否为空,为空则把它们指向新节点,反之,把新节点链在尾部,并更新尾部,最后插入完把size++。

1.3出队列

void QueuePop(Queue* qq)//出队列
{assert(qq);assert(!QueueEmpty(qq));if (qq->head->next == NULL){free(qq->head);qq->head = qq->tail = NULL;}else{QNode* del = qq->head;qq->head = qq->head->next;free(del);}qq->size--;}

出队列是进行头删,除了断言qq是否为空,还要判断整个队列是否为空,空的时候不能删。头删如果只有一个结点,直接free,并把头和尾置空;否则先把头部存在del指针变量中,再更新头结点,释放del。最后size个数–。

1.4取队首元素

QDataType QueueFront(Queue* qq)//取队列首元素
{assert(qq);assert(!QueueEmpty(qq));return qq->head->data;}

取队首也需要断言,直接返回头部指向的数据。

1.5取队尾元素

QDataType QueueBack(Queue* qq)//取队列尾元素
{assert(qq);assert(!QueueEmpty(qq));return qq->tail->data;}

取队尾直接返回尾部指向的数据。

1.6返回队列个数

int QueueSize(Queue* qq)//返回队列个数
{assert(qq);return qq->size;
}

返回队列个数,返回qq指向的个数。

1.7判断是否为空

bool QueueEmpty(Queue* qq)//判断是否为空
{assert(qq);return qq->head == NULL &&qq->tail == NULL;}

判断是否为空,直接返回head和tail同时为空,如果同时为空返回true,有一个不为空返回false。

1.8销毁队列

void QueueDestroy(Queue* qq)//销毁队列
{assert(qq);QNode* cur = qq->head;while (cur){QNode* del = cur;cur = cur->next;free(del);}qq->head = qq->tail = NULL;qq->size = 0;
}

队列销毁需要把头指针存到cur中,用while循环,把cur存到del指针变量中,更新cur,并释放掉del。最后把头指针和尾指针同时置为空,size置0。

(三)队列实现代代码

(1)Queue.c

#include"Queue.h"
void QueueInit(Queue* qq)//队列初始化
{assert(qq);qq->head = qq->tail = NULL;qq->size = 0;}void QueuePush(Queue* qq, QDataType x)//入队列
{assert(qq);QNode* newnode = (QNode*)malloc(sizeof(QNode));if (newnode == NULL){perror("malloc fail");}newnode->data = x;newnode->next = NULL;if (qq->tail == NULL){qq->head = qq->tail = newnode;}else{qq->tail->next = newnode;qq->tail = qq->tail->next;}qq->size++;
}void QueuePop(Queue* qq)//出队列
{assert(qq);assert(!QueueEmpty(qq));if (qq->head->next == NULL){free(qq->head);qq->head = qq->tail = NULL;}else{QNode* del = qq->head;qq->head = qq->head->next;free(del);}qq->size--;}QDataType QueueFront(Queue* qq)//取队列首元素
{assert(qq);assert(!QueueEmpty(qq));return qq->head->data;}QDataType QueueBack(Queue* qq)//取队列尾元素
{assert(qq);assert(!QueueEmpty(qq));return qq->tail->data;}int QueueSize(Queue* qq)//返回队列个数
{assert(qq);return qq->size;
}bool QueueEmpty(Queue* qq)//判断是否为空
{assert(qq);return qq->head == NULL &&qq->tail == NULL;}void QueueDestroy(Queue* qq)//销毁队列
{assert(qq);QNode* cur = qq->head;while (cur){QNode* del = cur;cur = cur->next;free(del);}qq->head = qq->tail = NULL;qq->size = 0;
}

(2)Queue.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
typedef int QDataType;
typedef struct QueueNode
{QDataType data;struct QueueNode* next;
}QNode;
typedef struct Queue
{QNode* head;QNode* tail;int size;
}Queue;
void QueueInit(Queue* qq);//队列初始化void QueuePush(Queue* qq,QDataType x);//入队列void QueuePop(Queue* qq);//出队列QDataType QueueFront(Queue* qq);//取队列首元素QDataType QueueBack(Queue* qq);//取队列尾元素int QueueSize(Queue* qq);//返回队列个数bool QueueEmpty(Queue* qq);//判断是否为空void QueueDestroy(Queue* qq);//销毁队列

(3)test.c

#include"Queue.h"
void test()
{Queue q;QueueInit(&q);QueuePush(&q, 9);QueuePush(&q, 8);printf("%d ", QueueFront(&q));QueuePop(&q);QueuePush(&q, 7);QueuePush(&q, 6);printf("%d\n", QueueEmpty(&q));printf("%d\n", QueueFront(&q));printf("%d\n", QueueBack(&q));while (!QueueEmpty(&q)){printf("%d ", QueueFront(&q));QueuePop(&q);}printf("\n");printf("%d\n", QueueSize(&q));
}
int main()
{test();return 0;
}

(四)队列测试结果

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_224049.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

FITC标记的STAT1-ASON,绿色荧光素标记STAT1反义寡核苷酸,FITC-STAT1-ASON

产品名称&#xff1a;FITC标记的STAT1-ASON&#xff0c;绿色荧光素标记STAT1反义寡核苷酸 ​​​​​​​英文名称&#xff1a;FITC-STAT1-ASON STAT1 是第一个被发现的 STATs 家族成员&#xff0c;其编码基因位于 2 号染色体上&#xff0c;由 750 个氨基酸残基组成&#xff…

随想录一刷Day55——动态规划

文章目录Day55_动态规划47. 判断子序列48. 不同的子序列Day55_动态规划 47. 判断子序列 392. 判断子序列 思路&#xff1a; 双指针很简单&#xff0c;O(n)O(n)O(n) 时间就能解决 这里还是用dp dp[i][j] 表示以 s[i - 1] 结尾的字符串和以 t[]i-1 为结尾的字符串的最大子序列长…

Linux篇【5】:Linux 进程概念(二)

目录 3.5、查看进程 3.6、通过系统调用接口获取正在进行的进程的标识符 3.7、通过系统调用接口创建子进程 - fork 初识 3.5、查看进程 [HJMhjmlcc ~]$ clear [HJMhjmlcc ~]$ pwd /home/HJM [HJMhjmlcc ~]$ ls [HJMhjmlcc ~]$ touch mytest.c [HJMhjmlcc ~]$ ls mytest.c [H…

基于51单片机的简易数字计算器Proteus仿真

资料编号&#xff1a;115 下面是相关功能视频演示&#xff1a; 115-基于51单片机的简易数字计算器Proteus仿真&#xff08;源码仿真全套资料&#xff09;功能说明&#xff1a; 该计算器系统51 系列的单片机进行的数字计算器系统设计&#xff0c;可以完成计算器的键盘输入&…

一看就会的Java方法

文章目录一、方法的定义和使用&#x1f351;1、为什么引入方法&#xff1f;&#x1f351;2、方法的定义&#x1f351;3、方法调用的执行过程&#x1f351;4、实参和形参的关系二、方法重载&#x1f351;1、为什么需要方法重载&#x1f351;2、方法重载的概念和特点&#x1f351…

用DIV+CSS技术设计的体育主题网站(足球介绍)

&#x1f389;精彩专栏推荐 &#x1f4ad;文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业&#xff1a; 【&#x1f4da;毕设项目精品实战案例 (10…

【面经】之小鼠喝药问题

题目 现在有 10 只小白鼠和 1000 支药水&#xff0c;1000 支药水中有且仅有一支药水有毒&#xff0c;如果小白鼠喝下毒药&#xff0c;那么毒发的时间是两小时。 现在只给你两小时的时间&#xff0c;请问如何用这 10 只小白鼠测出哪支药水有毒&#xff1f;&#xff08;忽略小白…

linux系统文件权限

目录 shell命令以及运行原理 具体体现(命令行解释器) Linux权限的概念 Linux下有两种用户&#xff1a;超级用户&#xff08;root&#xff09;、普通用户 su指令 Linux权限管理方面 文件访问者的分类&#xff08;人&#xff09; 为什么要有所属组&#xff1f; 文件属性…

STM32 Bootloader开发记录 2

在《stm32 bootloader开发记录.md》文档中&#xff0c;已经实现了Bootloader下的升级功能。可以在Bootloader启动时&#xff0c;进入升级模式&#xff0c;使用串口传输数据&#xff0c;来下载固件到flash中。 但是&#xff0c;在实际应用中&#xff0c;一般是在应用运行过程中…

基于单片机的指纹门禁设计

功能&#xff1a; 研究内容&#xff1a;本课题以单片机为核心采用C语言来开发一指纹电子密码锁。系统拟在Altium Designer9开发平台上设计原理图&#xff0c;并绘制PCB并制成单片机开发板&#xff0c;然后根据原理图将相关元器件焊接到开发板上。软件部分在Keil uVision4开发…

餐饮美食网页设计(HTML+CSS+JavaScript)

&#x1f380; 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业…

目标检测算法——自动驾驶开源数据集汇总2(附下载链接)

>>>深度学习Tricks&#xff0c;第一时间送达<<< 目录 一、Highway Driving 二、Mapillary Vistas 三、Cityscapes 四、CamVid >>>一起交流&#xff01;互相学习&#xff01;共同进步&#xff01;<<< 近期&#xff0c;小海带在空闲之余…

【FPGA】FPGA实现IIC协议读写EEPROM(二) -----EEPROM读写控制模块实现

EEPROM读写控制模块实现一、模块功能分析二、输入/输出信号三、EEPROM读写控制模块状态机四、EEPROM读写控制模块实现五、仿真测试写在前面&#xff1a; FPGA实现IIC协议读写EEPROM相关文章&#xff1a; IIC通信协议 【FPGA】FPGA实现IIC协议读写EEPROM&#xff08;一&#xff…

Kafka消费者之相关参数及分区分配再平衡

一、消费者重要参数 深刻的理解这些参数有利于大家在面对自己的项目场景上对配置文件有更好的把握&#xff01; 参数名称描述bootstrap.servers向 Kafka 集群建立初始连接用到的 host/port 列表。key.deserializer 和value.deserializer指定接收消息的 key 和 value 的反序列…

Spring--基于注解管理bean

基于注解管理bean 实验一&#xff1a;标记与扫描 注解 和 XML 配置文件一样&#xff0c;注解本身并不能执行&#xff0c;注解本身仅仅只是做一个标记&#xff0c;具体的功能是框架检测到注解标记 的位置&#xff0c;然后针对这个位置按照注解标记的功能来执行具体操作。 本质…

【ASM】字节码操作 工具类与常用类 asm-utils 与 asm-commons

1.概述 本章节主要是对 ASM中的 工具类与常用类 ,包asm-utils 与 asm-commons 两个包中的一些类进行讲解的介绍。 2. asm-util 在asm-util.jar当中,主要介绍CheckClassAdapter和TraceClassVisitor类。在TraceClassVisitor类当中,会涉及到Printer、ASMifier 和Textifier类。…

Vue中 引入使用 element-resize-detector 监听 Dom 元素 宽度、高度 变化

1. 前言 很多做pc端平台的小伙伴都遇到过这样一个问题&#xff1a;在做侧边栏菜单时会有一个收缩和展开的一个功能&#xff0c;在伸缩的过程中右边的页面的宽度就会随之改变。我上网查了查 &#xff0c;也动手试了试 window.onresize ()>{}。却不尽人意&#xff0c;因为它…

进程管理命令 动态监控进程 rpm yum

学习视频:074_韩顺平Linux_服务管理(2)_哔哩哔哩_bilibili 目录 进程管理命令基本介绍 PS命令 显示系统执行的进程 终止进程kill和killall 查看进程树pstree 服务管理 服务管理 打开或者关闭指定端口 动态监控进程 监控网络状态 …

数字IC手撕代码-XX公司笔试真题(脉冲密度调制)

前言&#xff1a; 本专栏旨在记录高频笔面试手撕代码题&#xff0c;以备数字前端秋招&#xff0c;本专栏所有文章提供原理分析、代码及波形&#xff0c;所有代码均经过本人验证。 目录如下&#xff1a; 1.数字IC手撕代码-分频器&#xff08;任意偶数分频&#xff09; 2.数字I…

nginx之https加密网站

目录 一、密钥算法 二、SSL虚拟主机 一、密钥算法 常见密钥算法&#xff1a; 对称加密&#xff1a;AES、DES 非对称加密&#xff1a;RSA、DSA 【注】对称加密的加密和解密使用的是同一把钥匙&#xff0c;非对称加密的加密和解密使用的不是一把钥匙&#xff0c;在对网…