数据结构(顺序队列——c语言实现)

队列的概念:

       队列是限制在两端进行插入和删除操作的线性表,允许进行存入的一端称为“队尾”,允许进行删除操作的一端称为“队头”。当线性表中没有元素时,称为“空队”。特点:先进先出(FIFO)。

规定一:front指向队头元素的前一个位置;rear指向队尾元素所在位置。

规定二:front指向队头元素的位置;rear指向队尾元素的下一个位置。

        在队列操作过程中,为了提高效率,以调整指针代替队列元素的移动,并将数组作为循环队列的操作空间。

       为了区别空队和满队,满队元素个数比数组元素个数少一个。

两个规定二选其一遵守,要么遵守一,要么遵守二。

顺序队列的优缺点:

优点

  1. 存储效率高

           顺序队列使用连续的内存空间,内存利用率较高,不会有像链表那样的指针开销。
  2. 访问速度快

           由于元素在内存中是连续存储的,顺序队列在访问元素时具有较高的缓存命中率,因此访问速度较快。
  3. 实现简单

           顺序队列的实现相对简单,通常只需要一个数组和两个指针(front 和 rear)来管理队列的头和尾。
  4. 空间预分配

           可以在创建队列时预先分配一个固定大小的数组,避免了在队列动态增长时频繁的内存分配和释放操作。

缺点

  1. 固定大小

           顺序队列的大小在创建时是固定的,如果队列满了,就不能再插入新元素,除非进行额外的扩容操作。扩容操作通常涉及复制整个数组到新的内存空间,这可能导致性能下降。
  2. 假溢出

           在顺序队列中,即使数组还有空闲空间,如果所有的空间都在队列的前面(已经被已删除的元素占用),但后面的空间还未使用,也会导致队列无法插入新元素,这就是所谓的“假溢出”问题。
  3. 内存碎片

           在频繁的插入和删除操作后,顺序队列可能会产生内存碎片,导致内存使用效率下降。虽然这通常不是主要问题,但在某些情况下需要考虑。
  4. 扩展性较差

           如果队列的大小需要频繁变化,顺序队列的扩展性较差。相比之下,链表实现的队列(如链式队列)在动态扩展方面更具优势。

顺序队列

下面以遵守规则二为例来编写代码:

#ifndef _SEQQUEUE_H
#define _SEQQUEUE_H#include <stdio.h>
#include <stdlib.h>
#include <string.h>typedef char Type;
#define LEN 10
//约定1:LEN-1号下标的下一个为0号下标
//约定2:数组少存一个数据,最多只存LEN-1个元素
typedef struct{Type data[LEN];   //队列空间int front;        //队头的下标int rear;         //队尾下标+1
}queue;//创建
queue *create_queue();
//判空 空为0非空为1
int empty_queue(queue *q);
//判满 满为0不满为1
int full_queue(queue *q);
//入队enqueue
void enter_queue(queue *q,Type data);
//出队dequeue
Type delete_queue(queue *q);
//初始化
void init_queue(queue *q);
//回收
void free_queue(queue **q);#endif

       顺序队列同样是使用数组来存储队列中的元素,结构体里有三个成员,一个是存放队列元素的数组,另外两个是队列数组两个下标;数组的长度用一个宏定义的LEN来代表,之后要修改队列长度直接修改宏定义即可。

#include "seqqueue.h"//创建
queue *create_queue()
{queue *q = (queue *)malloc(sizeof(queue));if(NULL == q){perror("create queue malloc");return NULL;}q->front = q->rear = 0;return q;
}
//判空 空为0,非空为1
int empty_queue(queue *q)
{if(NULL == q){puts("queue is NULL");return -1;}if(q->front == q->rear){puts("queue is empty");return 0;}elsereturn 1;
}
//判满 满为0,不满为1
int full_queue(queue *q)
{if(NULL == q){puts("queue is NULL");return -1;}if(q->front == (q->rear+1)%LEN){puts("queue is full");return 0;}elsereturn 1;
}
//入队enqueue
void enter_queue(queue *q,Type data)
{if(1 == full_queue(q)){
#if 1q->rear = q->rear%LEN; //让10变为0q->data[q->rear] = data;q->rear++;
#elseq->data[q->rear] = data;q->rear = (q->rear+1)%LEN;
#endif}
}
//出队dequeue
Type delete_queue(queue *q)
{if(1 == empty_queue(q)){Type t = q->data[q->front];q->front = (q->front+1)%LEN;return t;}
}
//初始化
void init_queue(queue *q)
{if(NULL == q){puts("queue is NULL");return;}q->rear = q->front;
}
//回收
void free_queue(queue **q)
{if(NULL == *q){puts("queue is NULL");return;}free(*q);*q = NULL;
}

创建: queue *create_queue()

       在创建顺序队列的时候也是在堆区开辟空间,在这里一开始队列为空,所以这时候尾的下标应该为-1,头的下标为0,但是我们规定rear为尾+1,因此在创建完结构体之后,给front和rear都赋值为0即可。

判空:int empty_queue(queue *q)

       在前面的分析中我们已经知道了队列为空的时候front和rear的值应该相同,因此在这里只需要判断结构体里的front和rear是否相等就能判断队列是否为空,空函数返回0,非空函数返回1。

判满:int full_queue(queue *q)

       在判满这里分析中也提到了,为了避免判空和判满发生冲突,所以数组最多存储LEN-1个元素,但是除此之外,为了提高利用率,我们把数组作为循环队列来操作;在这里就会出现一个问题,那就是当队列放满,数据存储刚好到数组倒数第二位时,rear指向的就是数组最后一个元素的位置,这时rear+1就会超出数组下标的范围,此时front为数组第一个元素的下标,所以为了让rear+1与front相等,我们对rear+1进行除以LEN取余操作,这样当rear+1=LEN时,取余正好为0,就是第一个元素的下标;故判满的条件就为(q->front == (q->rear+1)%LEN

入队:void enter_queue(queue *q,Type data)

       入队是在队尾的位置进行操作,也就是下标为rear的位置,但是在入队操作之前我们需要做一个判断,如果队列是满的那就不能再进行入队操作,只有队列不满才能进行入队操作;在入队之前我们需要保证rear的值不超过LEN-1,也就是当数组最后一个位置放满尾就应该移动到数组开头,所以在这里就先让rear取余LEN,再把data放入队列,最后让rear往后移动一位即可。

出队:Type delete_queue(queue *q)

       出队我们需要拿到出队的这个成员的值,因此函数返回值为Type类型,出队之前同样也需要进行一个判断,如果队列是空的,那就不能进行出队操作;当队列非空时,操作队头进行出队操作,将要出队的元素用一个变量t保存,然后让队头下标front往后移动一位,在这里为了防止超出数组操作范围,也需要对front进行取余,保证它在0~9之间;最后再将t返回即可。

初始化:void init_queue(queue *q)

       初始化的意思也就是将队列恢复到最开始空的时候,对于顺序队列,它为空的时候是front=rear,因为是数组,因此赋值可以覆盖,所以初始化我们只需要让结构体里的队头下标front和rear相等即可,这样队列相当于就是空队列,后续赋值会直接覆盖,不会造成别的影响。

回收:void free_queue(queue **q)

       在这里传参同样需要传指针的地址,因为存数据的数组在结构体里面,故在回收的时候不需要进行初始化,直接将结构体空间回收,再将指针指向NULL即可。

测试(主函数)

#include "seqqueue.h"
int main(int agrc,char *agrv[])
{queue *q = create_queue();printf("空为0:%d\n",empty_queue(q));puts("A,B,C,D入队");enter_queue(q,'A');enter_queue(q,'B');enter_queue(q,'C');enter_queue(q,'D');puts("出队");printf("%c\n",delete_queue(q));printf("%c\n",delete_queue(q));printf("判空:%d\n",empty_queue(q));puts("初始化");init_queue(q);printf("判空:%d\n",empty_queue(q));puts("回收");free_queue(&q);printf("%p\n",q);return 0;
}

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

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

相关文章

Vulnhub靶场 Jangow: 1.0.1 练习

目录 0x00 准备0x01 主机信息收集0x02 站点信息收集0x03 漏洞查找与利用1. 命令执行2. 反弹shell3. 提权4. 补充4.1 其他思路4.2 问题 0x04 总结 0x00 准备 下载链接&#xff1a;https://download.vulnhub.com/jangow/jangow-01-1.0.1.ova 介绍&#xff1a; Difficulty: easy…

Fakelocation Server服务器/专业版 Centos7

前言:需要Centos7系统 Fakelocation开源文件系统需求 Centos7 | Fakelocation | 任务一 更新Centos7 &#xff08;安装下载不再赘述&#xff09; sudo yum makecache fastsudo yum update -ysudo yum install -y kernelsudo reboot//如果遇到错误提示为 Another app is curre…

【Ubuntu24.04】服务部署(虚拟机)

目录 0 背景1 安装虚拟机1.1 下载虚拟机软件1.2 安装虚拟机软件1.2 安装虚拟电脑 2 配置虚拟机2.1 配置虚拟机网络及运行初始化脚本2.2 配置服务运行环境2.2.1 安装并配置JDK172.2.2 安装并配置MySQL8.42.2.3 安装并配置Redis 3 部署服务4 总结 0 背景 你的服务部署在了你的计算…

深入解析 EasyExcel 组件原理与应用

✨深入解析 EasyExcel 组件原理与应用✨ 官方&#xff1a;EasyExcel官方文档 - 基于Java的Excel处理工具 | Easy Excel 官网 在日常的 Java 开发工作中&#xff0c;处理 Excel 文件的导入导出是极为常见的需求。 今天&#xff0c;咱们就一起来深入了解一款非常实用的操作 Exce…

Java爬虫:获取商品详情的实践之旅

在当今这个信息爆炸的时代&#xff0c;数据的价值日益凸显。对于电商行业来说&#xff0c;商品详情的获取尤为重要&#xff0c;它不仅关系到产品的销售&#xff0c;还直接影响到用户体验。传统的人工获取方式耗时耗力&#xff0c;而自动化的爬虫技术则提供了一种高效解决方案。…

C# 数据结构之【树】C#树

以二叉树为例进行演示。二叉树每个节点最多有两个子节点。 1. 新建二叉树节点模型 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace DataStructure {class TreeNode{public int Data { get;…

单片机_简单AI模型训练与部署__从0到0.9

IDE&#xff1a; CLion MCU&#xff1a; STM32F407VET6 一、导向 以求知为导向&#xff0c;从问题到寻求问题解决的方法&#xff0c;以兴趣驱动学习。 虽从0&#xff0c;但不到1&#xff0c;剩下的那一小步将由你迈出。本篇主要目的是体验完整的一次简单AI模型部署流程&#x…

【Spiffo】环境配置:VScode+Windows开发环境

摘要&#xff1a; 在Linux下直接开发有时候不习惯快捷键和操作逻辑&#xff0c;用Windows的话其插件和工具都更齐全、方便&#xff0c;所以配置一个Windows的开发环境能一定程度提升效率。 思路&#xff1a; 自己本地网络内远程连接自己的虚拟机&#xff08;假定用的是虚拟机…

使用eclipse构建SpringBoot项目

我这里用eclipse2018版本做演示&#xff0c;大家有需要的可以下载Eclipse Downloads | The Eclipse Foundation 1.打开eclipse&#xff0c;选择存放代码的位置 2.选择 file >> new >> project >> 选择springboot文件下的 spring starter project 2.这里选择N…

C++ 日期计算器的实现(运算符重载)

目录 一、前言 二、正文 1.1 Date类框架 1.2 两个日期间的直接赋值 1.3判断两个日期是否相同 1.4判断两个日期是否不同 1.5日期加天数 1.6日期天数 1.7日期的前置和后置 1.8日期减天数 1.9自身日期减天数 2.1自身日期前置--和后置-- 2.2两个日期的差值为差距天数 2…

LLM的原理理解6-10:6、前馈步骤7、使用向量运算进行前馈网络的推理8、注意力层和前馈层有不同的功能9、语言模型的训练方式10、GPT-3的惊人性能

目录 LLM的原理理解6-10: 6、前馈步骤 7、使用向量运算进行前馈网络的推理 8、注意力层和前馈层有不同的功能 注意力:特征提取 前馈层:数据库 9、语言模型的训练方式 10、GPT-3的惊人性能 一个原因是规模 大模型GPT-1。它使用了768维的词向量,共有12层,总共有1.…

如何使用 Python 开发一个简单的文本数据转换为 Excel 工具

目录 一、准备工作 二、理解文本数据格式 三、开发文本数据转换为Excel工具 读取CSV文件 将DataFrame写入Excel文件 处理其他格式的文本数据 读取纯文本文件: 读取TSV文件: 四、完整代码与工具封装 五、使用工具 六、总结 在数据分析和处理的日常工作中,我们经常…

太通透了,Android 流程分析 蓝牙enable流程(应用层/Framework/Service层)

零. 前言 由于Bluedroid的介绍文档有限&#xff0c;以及对Android的一些基本的知识需要了(Android 四大组件/AIDL/Framework/Binder机制/JNI/HIDL等)&#xff0c;加上需要掌握的语言包括Java/C/C等&#xff0c;加上网络上其实没有一个完整的介绍Bluedroid系列的文档&#xff0…

李继刚:提示词(Prompt)的本质是表达的艺术

看了李继刚在 AI 创新者大会的演讲《提示词的道与术》&#xff0c;收获很大&#xff0c;我分享一下学习笔记。  李继刚&#xff1a;提示词&#xff08;Prompt&#xff09;的本质是表达的艺术 一、提示词的本质是表达 本意、文意和解意的概念&#xff1a; 本意&#xff1a;指…

复古风格渐变褪色人像旅拍Lr调色教程,手机滤镜PS+Lightroom预设下载!

调色教程 这种调色风格旨在通过调整色彩和光影&#xff0c;为人像旅拍照片赋予复古的氛围和艺术感。渐变褪色效果增添了一种时光沉淀的感觉&#xff0c;使照片仿佛来自过去的岁月。 预设信息 调色风格&#xff1a;复古风格预设适合类型&#xff1a;人像&#xff0c;街拍&…

Android Studio更改项目使用的JDK

一、吐槽 过去&#xff0c;在安卓项目中配置JDK和Gradle的过程非常直观&#xff0c;只需要进入Android Studio的File菜单中的Project Structure即可进行设置&#xff0c;十分方便。 原本可以在这修改JDK: 但大家都知道&#xff0c;Android Studio的狗屎性能&#xff0c;再加…

字节青训营开课啦

系列文章目录 文章目录 系列文章目录一、字节青训营是什么&#xff1f;二、你将获得三、入营条件四、课程简介1.前端2.后端3.大数据 五、报名 一、字节青训营是什么&#xff1f; 青训营是字节跳动技术团队发起的技术系列培训&人才选拔项目&#xff1b;面向高校在校生&…

医药企业的终端市场营销策略

近年来&#xff0c;随着医药行业的快速发展&#xff0c;终端市场逐渐成为企业竞争的关键领域。在政策趋严、市场环境变化以及数字化转型的大背景下&#xff0c;医药企业如何在终端市场中立于不败之地&#xff1f;本文结合我们在医药数字化领域的经验&#xff0c;为大家剖析终端…

养老院管理系统+小程序项目需求分析文档

智慧综合养老服务平台是以业务为牵引、场景为驱动&#xff0c;围绕“老人”业务域&#xff0c;持续沉淀和打磨形成适应不同养老业务发展需要的业务能力&#xff0c;推动业务模式升级&#xff0c;为养老服务提供数字化解决方案&#xff0c;并依托实体站点与养老机构实现线上线下…

SpringBoot集成ES(ElasticSearch)

1.导入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-elasticsearch</artifactId> </dependency>导入依赖后&#xff0c;注意在依赖中查看对应的版本是否与本机ES对应 2.创建配置并…