常用数据结构 ——— 队列(环形队列和顺序队列)

news2025/1/10 16:20:17

目录

一、队列简介

二、顺序队列

三、环形队列

四、环形队列代码

1、队列结构体

2、队列初始化

3、判断队列是否为满

4、判断队列是否为空

5、将数据插入到队列中

6、读取队列中的数据

7、释放队列空间

8、功能测试


一、队列简介

        队列只允许在队列头(front)进行删除操作,在队列尾(rear)进行插入操作,当队列中没有元素时,称为空队列在队列中插入元素称为入队,从队列中删除元素称为出队。因为队列只允许在尾端插入,在头端删除,所以只有最早进入队列的元素才能最先从队列中删除,即队列有先进先出的特点。

二、顺序队列

        即仅在队头进行删除在队尾进行插入,可用地址连续的存储单元依次存放队列中的数据,比如数组。队头和队尾的位置是变化的,所以要设置头、尾指针。  

        初始化时的头尾指针,均置为 0。 当头尾指针相等时队列为空或者为满,在非空队列里,头指针始终指向队头元素,尾指针始终指向队尾元素的下一位置。

        由队列的原理可以将头指针当做读操作,将尾指针当做写操作,即在尾端插入数据就是写入队列,在头端删除数据就是将队列中的数据读出,这样好理解点。

刚开始头指针和尾指针都在同一位置

 当队列入队时,尾指针加一,头指针保持不变,a1入队,尾指针rear+1指向下一个地址空间,即尾指针始终指向队尾的下一地址,如a4入队后,尾指针rear+1 。

队列出队时,尾指针保持不变,头指针依次加1,由先进先出原则,a1先入队,则a1先被读走,然后front+1,指向了 a2,a2删除后,front+1指向了 a3。

当a5删除后,头指针和尾指针的指向又相同相等了,即说明队列中的数据 已经全部读走

在顺序队列中,当尾指针已经指向了队列的最后一个位置的下一位置时,如果再有元素入队,就会发生“溢出”此时队列中已经填满了数据,头指针还在开始位置。

顺序队列的 “假溢出” :即队列的存储空间并未填满,却发生了溢出。

        比如 rear 现在指向了最后一个位置的下一位置,按照上面所说此时队列已经被填满,如果再有元素入队,就会发生“溢出”,但这是在头指针没有移动的前提下,如果之前队列头也删除了一些元素,那么队列头指针经过n次的 +1 之后,会遗留下了很多空地址,但是顺序队列就会认为再有元素入队,就溢出,即出现 “假溢出” 现象,这是不合理的,故出现了环形队列

三、环形队列

        环形队列的使用场景还是挺多的,比如要将单片机一些模块采集的数据连续上传到PC端,这里就可以用到环形队列,即将采集的数据放到队列中去,再将队列中的数据读出上传到PC端,这里为什么不直接将采集端和上传端直接相连呢?因为采集数据的速度和上传数据的速度是不知道的,如果直接相连会出错。

        环形队列的原理就是将新元素加入到第一个位置上,构成类似于一个环一样的队列,入队和出队还是按照“先进先出”的原则进行,环形队列的空间利用率高。

        环形队列解决了顺序队列 “假溢出” 的现象,但是又出现新的问题,即怎么判断队列是否为空,如果单用 rear = front 判断空或满显然是不行的,比如

         此时两种情况都是 rear = front 的情况,在环形队列中,当队列满了之后 rear + 1会指向第一个地址,即出现了  rear = front 的情况。

        一般判断队列空的条件是 rear = front, 通常少用一个元素的储存空间用来判断队列是否满,即在入队前测试尾指针加 1 后是否等于头指针,若相等则认为队满。

        或者当用数组array[len]来表示队列时,则可用(rear - front)的差和数组长度 len 进行比较,如果相等则说明队列已满,(在下面说明这种方法)。

四、环形队列代码

 W等于尾指针rear,R等于头指针front,相当于写操作读操作

1、队列结构体

/*队列结构体*/
typedef struct ring_buff{

	int array[len]; 
	int W;
    int R;
}*ring;

2、队列初始化

/*队列初始化*/
struct ring_buff *fifo_init(void)
{
	struct ring_buff *p = NULL;
	p = (struct ring_buff *)malloc(sizeof(struct ring_buff));
    if(p == NULL){
		printf("malloc error\n");
		return -1;
    }else{
		p->W = 0;
        p->R = 0;
        memset(p->array,0,sizeof(p->array));
        return p;
    }
}

将队列相应空间清零。

3、判断队列是否为满

/*判断是否满*/
int get_ring_buff_fullstate(struct ring_buff *P_ring_buff)
{
	if((P_ring_buff->W - P_ring_buff->R) == len){
		return 1;
    }else{
			return 0;
    }
}

        如上图所示,array[6] 的存储空间为array[0] - array[5],即6个数,当RW等于0时指向array[0],存入数据后W++,指向下一个地址,当W将array[5]存入数据后会自加一,此时W = 6,R = 0,下一次判断队列是否满时条件成立,即W  -   R  =  len,len为数组长度6

4、判断队列是否为空

/*判断是否为空*/
int get_ring_buff_emptystate(struct ring_buff *P_ring_buff)
{
	if(P_ring_buff->W == P_ring_buff->R){
		return 1;
    }else{
		return 0;
    }
}

当头指针和尾指针相等时判断为空

5、将数据插入到队列中

/*插入数据*/
int ring_buff_insert(struct ring_buff *P_ring_buff,int data)
{
	if(P_ring_buff == NULL){        
    	printf("insert P_ring_buff error\n");
		return -1;
    }
    /*判断队列是否满*/
    if(get_ring_buff_fullstate(P_ring_buff) == 1){
		printf("full\n");
		return -1;
    }
	P_ring_buff->array[P_ring_buff->W%len] = data;
    P_ring_buff->W++;
    return 0;
}

W = {0、1、2、3、4、5 },则array[W%len] = {0、1、2、3、4、5},刚好依次对应,当W = 6时,6 % 6 = 0,又从array[0]开始存储数据,这也是环形队列的关键,即形成闭环。

6、读取队列中的数据

/*读取数据*/
int ring_buff_get(struct ring_buff *P_ring_buff)
{
	int data = 0;
    if(P_ring_buff == NULL){
		printf("get P_ring_buff error\n");
		return -1;
    }
    data = P_ring_buff->array[P_ring_buff->R%len];
    P_ring_buff->R++;
    return data;
}

读取数据和写入数据类似,也是当R = 6时,6 % 6 = 0,又从array[0]开始读取数据

7、释放队列空间

/*销毁*/
int ring_buff_destory(struct ring_buff *P_ring_buff)
{
	if(P_ring_buff == NULL){
		printf("destory P_ring_buff error\n");
		return -1;
    }
	free(P_ring_buff);
    return 0;
}

8、功能测试

#define len 6
int main() 
{
	int i;
	int getData = 0;
/*初始化队列*/
    ring Pt_ring_buff = fifo_init(); 
/*向队列中写数据,即0 - 5*/
	for(i = 0;i < 6;i++){
		ring_buff_insert(Pt_ring_buff,i);
    }
/*将写入的数据读出三个,由先进先出原则应该读的0 - 2*/
    for(i = 0;i < 3;i++){
		printf("  %d",ring_buff_get(Pt_ring_buff));
    }
    printf("\n");
/*再写入三个数据,此时写入的数据应该在上面的数据之后,即在5之后*/
    for(i = 8;i < 11;i++){
    ring_buff_insert(Pt_ring_buff,i);
    }
/*读取队列中的数据,此时再次读取队列中的数据时,0 - 2 已经被读走,所以应该是从3、4、5、8、9、10、3 ...循环读取十个数据*/    
    for(i = 0;i < 10;i++){
		printf("  %d",ring_buff_get(Pt_ring_buff));
    }
	ring_buff_destory(Pt_ring_buff);   //释放空间
	printf("\n");
    system("pause");
    return 0;
}

输出结果

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

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

相关文章

_Linux(共享内存)

文章目录0. 共享内存1. 共享内存示意图2. 共享内存函数2.1 shmget函数2.2 shmat函数2.3 shmdt函数2.4 shmctl函数2.5 查看共享内存指令2.6 删除共享方法2.6.1 指令删除2.6.2 代码删除3. 实例代码3.0 log.hpp打印日志信息3.1 comm.hpp(shmServer.cc和shmClicent.cc共有文件)3.2 …

Java 17的这些新特性,Java迈入新时代

前言 2021年9月14日Java 17发布&#xff0c;作为新时代的农民工&#xff0c;有必要了解一下都有哪些新东西。 Java 17是Java 11以来又一个LTS&#xff08;长期支持&#xff09;版本&#xff0c;Java 11 和Java 17之间发生了那些变化可以在OpenJDK官网找到JEP&#xff08;Java…

Unity中用Natrue Renderer做自己的地形Terrain.

效果图 一、下载与导入Nature Renderer Nature Renrderer是个强大的插件&#xff0c;它本身就可以作为地形编辑的工具取代Unity的地形细节和树木的渲染系统。 nature-renderer官网 1.下载链接 推荐&#xff08;已经购买的许可证&#xff0c;可直接使用&#xff09;&#xf…

设计原则和设计模式01

一&#xff1a;软件设计原则 1.单一职责原则&#xff1a; 有且只有一个原因引起类的变化(类或者接口的职责单一化) 2.里氏替换原则&#xff1a; 子类可以扩展父类的功能,但不能改变父类原有的功能 3.依赖倒置原则&#xff1a; 1.高层模块不应该依赖于底层模块&#xff0c…

Java注解

Java注解&#xff08;Annotation&#xff09; Java 注解&#xff08;Annotation&#xff09;又称 Java 标注&#xff0c;是 JDK5.0 引入的一种注释机制。 注解也叫元数据&#xff0c;一种代码级别的说明&#xff0c;说明程序的是给计算机看的&#xff0c;与类&#xff0c;接口…

事件绑定(onsubmit)表单提交

事件绑定(onsubmit)表单提交 学习路线&#xff1a;JavaScript基础语法&#xff08;输出语句&#xff09;->JavaScript基础语法&#xff08;变量&#xff09;->JavaScript基础语法&#xff08;数据类型&#xff09;->JavaScript基础语法&#xff08;运算符&#xff09…

Python笔记 · Python语言的“动态性”

尽管对于Python程序员来说已经司空见惯&#xff0c;但是当那些从非动态语言转过来的程序员初次看到形如self.xxxxxx的语句就是在定义对象属性时往往会感到“离奇”&#xff1a;一个未经声明的&#xff08;类似private int a;那样&#xff09;变量&#xff0c;直接从self中“点”…

java知识回顾笔记(对象、反射、内省、实例、父类、构造方法、封装、泛型、super())

类&对象 在创建了一个类时&#xff0c;只声明但不赋值&#xff0c;其默认值为&#xff1a; 理解下图含义&#xff0c;即可理解对象和类&#xff1a; 实例 对象又被称为实例&#xff0c;实例变量被创建时&#xff0c;系统默认会赋值&#xff0c;例如&#xff1a; Studen…

什么知识库工具适合小团队?看看文档管理系统+NAS的最新解决方案

编者按&#xff1a;还在为团队选那款网盘而发愁吗?试试文档管理系统和NAS结合吧&#xff0c;高效率低成本&#xff0c;适合小团队。 关键词&#xff1a;免维护&#xff0c;免安装&#xff0c;大容量&#xff0c;在线编辑&#xff0c;文档共享&#xff0c;数据安全 对于企业或…

LeetCode-66-加一

1、从后向前遍历 我们可以从后向前遍历数组&#xff0c;针对不同的情况进行操作&#xff1a;1、若当前数字不为9&#xff0c;则我们直接将数字的值加一并返回即可&#xff1b;2、若当前数字为9&#xff0c;我们将当前数字置为0并对前一位执行加一操作&#xff1b;3、若所有数字…

后端接口时通时不通,团队全链路排查实战

背景&#xff1a; 1 最近团队做了一套系统&#xff0c;已经临近上线了&#xff1b; 2 后端的服务和前端的代码都是新写的&#xff0c;两边的服务器&#xff0c;数据库也都是新申请的&#xff1b; 3 本来测试的时候用的测试服务器&#xff0c;一切都挺好的&#xff0c;但部署到线…

基于分发与计算的GRTN全球实时传输网络

一张能同时满足「分发」与「计算」需求的网。 从直播趋势看「分发」与「计算」 阿里云直播产品架构图中&#xff0c;主要分为端和云两个部分&#xff1a;在端侧&#xff0c;主要包含推流端和播放端&#xff1b;在云侧&#xff0c;一是基于分布式节点构建的传输网&#xff0c;二…

mosquitto部署mqtt broker 并测试订阅与发布

mosquitto部署mqtt broker 并测试订阅与发布 1&#xff0c;MQTT协议介绍 MQTT(消息队列遥测传输)是ISO 标准(ISO/IEC PRF 20922)下基于发布/订阅范式的消息协议。它工作在 TCP/IP协议族上&#xff0c;是为硬件性能低下的远程设备以及网络状况糟糕的情况下而设计的发布/订阅型…

如何使用phpstudy在服务器上发布Discuz_X3.4_SC_UTF8_20220811和zzcms2023

Web-ZZCMS&#xff1a; 源码下载&#xff1a;http://www.zzcms.net/about/6.html Web-Discuz&#xff1a; 源码下载&#xff1a;https://www.discuz.net/ 首先安装phpstudy&#xff0c;傻瓜式安装教程&#xff0c;如果中途 遇到报错比如我遇到的是Apache未启动&#xff0c;80端…

Ansible自动化部署安装openGauss3.1企业版单机

文章目录一、背景二、环境准备三、具体实施步骤3.1、安装ansible3.2、配置主机清单3.3、测试主机连通性3.4、创建相关目录3.5、下载openGauss软件包到files目录3.6、创建变量文件3.7、创建安装时需要的xml模板3.8、创建任务文件四、执行自动化安装4.1、校验语法4.2、自动化安装…

CAD新手必练图形三

这次说的这个CAD新手必练图形&#xff0c;用到了CAD直线、圆、多边形、修剪和旋转等多个CAD功能命令&#xff0c;一起跟着操作一下吧 目标图形 操作步骤 1.使用CAD直线命令绘制一条垂直的直线和两条水平的直线&#xff0c;两条水平的直线之间的距离为15&#xff0c;并将他们修…

【CSDN竞赛第11期】编程竞赛总结

文章目录前言/背景题目与解题代码1. 圆小艺2. K皇把妹3. 筛选宝物4. 圆桌CSDN编程竞赛报名地址&#xff1a;https://edu.csdn.net/contest/detail/16 前言/背景 目前已经连续参加10次CSDN的编程竞赛了&#xff0c;这种比赛挺有意义&#xff0c;希望一直举办下去&#xff01; 题…

JDBC 访问数据库

文章目录一、实验目的二、实验要求三、实验内容1、设计数据库表2、定义实体类3、定义数据库连接类4、实现数据库的增删改查5、测试用例一、实验目的 理解 JDBC 的工作原理&#xff0c;掌握 JDBC 访问数据库&#xff1b; 掌握常见数据库 MYSQL &#xff1b; 二、实验要求 理解…

【华为上机真题 2022】字符串序列判定

&#x1f388; 作者&#xff1a;Linux猿 &#x1f388; 简介&#xff1a;CSDN博客专家&#x1f3c6;&#xff0c;华为云享专家&#x1f3c6;&#xff0c;Linux、C/C、云计算、物联网、面试、刷题、算法尽管咨询我&#xff0c;关注我&#xff0c;有问题私聊&#xff01; &…

中断系统中的设备树__中断号的演变与irq_domain

1 中断号与硬件相关_号码固定 当我们的系统中只有一两个中断控制器时&#xff0c;上面的方法很有用&#xff0c;可以给每一个中断预先确定好他的虚拟中断号&#xff0c;但是当中断控制器越来越多、当中断越来越多&#xff0c;上述方法(virq和hwirq固定绑定)有缺陷: a. 增加工作…