顺序队列----数据结构

news2025/1/16 18:03:59

队列的概念

队列,符合先进先出特点的一种数据结构,是一种特殊的线性表,但它不像线性表一样可以任意插入和删除操作,而是只允许在表的一端插入,也就是在队列的尾部进行插入;只允许在表的另一端进行删除,也就是在队列的头部进行删除。

以下的实现是顺序队列(存储空间在内存上是连续的队列)

队列的实现

队列的结构定义

#define MAX_SIZE 20            //队列的最大容量
typedef int DateElem;   

typedef struct _Queue
{
	DateElem date[MAX_SIZE];
	int head;				//头指针
	int tail;				//尾指针
}squeue;

队列的初始化

void InitQueue(squeue* sq)
{
	if (!sq) return;
	sq->head = 0;
	sq->tail = 0;
}

销毁(清空)队列

和队列的初始化有点类似哈哈。

bool DestoryQueue(squeue* sq)
{
	if (!sq) return false;

	sq->head = 0;
	sq->tail = 0;
	return true;
}

判满

bool IsFull(squeue* sq)
{
	if(!sq) return false;    //防御性编程

	if (sq->tail >= MAX_SIZE ) //每入队一个元素,sq->tail++,入队了MAX_SIZE个元素刚好sq->tail等于MAX_SIZE
	{
		return true;
	}
	else
	{
		return false;
	}
}

判空

可以从 队列的初始状态(空队列)入队后再出队 中找到判空的条件。

bool IsEmpty(squeue* sq)
{
	if (!sq) return false;

	if (sq->head == sq->tail) //两个指针之间没有元素
	{
		return true;
	}
	else
	{
		return false;
	}
}

入队

bool EnterQueue(squeue* sq, DateElem e)
{
	if (IsFull(sq))
	{
		cout << "无法插入元素"<<e<<",队列已满。" << endl;
		return false;
	}

	sq->date[sq->tail ] = e;  //尾部插入
	sq->tail++;               //尾指针指向下一块未被使用区域 
	return true;
}

出队

出队的方式有两种,一种是乾坤大挪移,每出队一个元素都要将后面所有的元素往前挪,十分浪费时间。另一种是舍弃空间来达到快速出队元素。

第一种

bool PopQueue(squeue* sq, DateElem* date)
{
	if (!sq || IsEmpty(sq)) return false;

	*date = sq->date[sq->head];         //返回出队的元素
	for (int i = sq->head + 1; i < sq->tail; i++)
	{
		sq->date[i - 1] = sq->date[i]; //从第二个结点开始,将第二个结点赋值给第一个结点……
	}
	sq->tail--;    //别忘记
	return true;
}

第二种

bool PopQueue2(squeue* sq,DateElem *date)
{
	if (!sq || IsEmpty(sq)) return false;

	//不是无限制的出队
	if (sq->head >= MAX_SIZE) return false;

	*date = sq->date[sq->head];
	sq->head++;
	return true;
}

本来头指针一直指向下标为 0 的地方,但是这种出队方式会导致头指针一直向后移动,出现“假溢出”,明明队列还有空间存储,可是却无法插入元素了。如下图:

造成了空间的浪费,不过在比赛时为了通过题目,这点空间浪费无所谓,使用顺序队列非常容易构建。

 打印队列

和链表的打印一样。

bool PrintQueue(squeue* sq)
{
	if (!sq) return false;

	for (int i = sq->head; i < sq->tail; i++)
	{
		printf("%d ", sq->date[i]);
	}
	return true;
}

获取队首元素

int GetHeadElem(squeue* sq)
{
	if (!sq || IsEmpty(sq)) return 0; //指针存在 或者 队列不为空
	
	return sq->date[sq->head];        //在队头不出队的清况下,返回队首元素
}

获取队列长度

int GetLength(squeue* sq)
{
	if (!sq) return 0;

	return sq->tail - sq->head; //最开始为空队列sq->tail - sq->head 为 0 ,依次类推得到长度
}

主函数

我已经写好了测试的方法,可以尽情调试看看代码的正确性。

int main(void)
{
	squeue* sq = new squeue;
	DateElem* s = new DateElem;
	InitQueue(sq);
	DateElem e = 0;

	int choose = -1;
	while (choose != 0)
	{
		cout << "1.入队" << endl
			<< "2.出队" << endl
			<< "3.打印队列" << endl
			<< "4.获取队首元素" << endl
			<< "5.获取队列长度" << endl
			<< "6.销毁队列" << endl
			<< "0.退出" << endl;
		cin >> choose;

		switch (choose)
		{
		case 1:
			cout << "请输入要入队的元素:";
			cin >> e;
			if (EnterQueue(sq, e))
			{
				cout << "入队成功" << endl;
			}
			else
			{
				cout << "入队失败" << endl;
			}
			break;
		case 2:
			if (PopQueue(sq, s))
			{
				cout << "出队的元素是:" << *s << endl;
			}
			else
			{
				cout << "出队失败" << endl;
			}
			break;
		case 3:
			cout << "队列中的元素是:";
			PrintQueue(sq);
			cout << endl;
			break;
		case 4:
			cout << "队首元素是:" << GetHeadElem(sq) << endl;
			break;
		case 5:
			cout << "队列的长度是:" << GetLength(sq) << endl;
			break;
		case 6:
			if (DestoryQueue(sq))
			{
				cout << "队列已销毁" << endl;
			}
			else
			{
				cout << "队列不存在" << endl;
			}
			break;
		case 0:
			cout << "退出成功" << endl;
			break;
		default:
			cout << "输入非法" << endl;
			break;

		}
	}


	return 0;
}

好了再见!

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

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

相关文章

Vue2之防抖_debounce封装函数v-debounce自定义指令(传参/不传)

目录 1、防抖 2、debounce - 封装函数 3、v-debounce 全局自定义指令 1、防抖 推荐文章 &#xff1a; https://blog.csdn.net/weixin_58099903/article/details/119902796 2、debounce - 封装函数 utils / tools.js /*** 函数防抖 是n秒后延迟执行&#xff0c;多用于页面scr…

js获取视频编码

一.背景 有些浏览器不支持某些视频的编码方式导致播放出现问题&#xff0c;这个时候要限制视频上传 二.插件 https://unpkg.com/mediainfo.js0.1.4/dist/mediainfo.min.js 三.完整html代码 <!DOCTYPE html> <html lang"en"> <head><meta ch…

【微服务保护】Sentinel 流控规则 —— 深入探索 Sentinel 的流控模式、流控效果以及对热点参数进行限流

文章目录 前言一、快速掌握 Sentinel 的使用1.1 什么是簇点链路1.2 Sentinel 的简单使用示例 二、Sentinel 流控模式2.1 直接模式2.2 关联模式2.3 链路模式 三、流控效果3.1 快速失败3.2 预热模式3.3 排队等待 四、对热点参数的流控4.1 热点规则4.2 热点规则演示 前言 微服务架…

flutter doctor检测环境,出现CocoaPods installed but not working

1. 安装flutter, 地址: 安装和环境配置 - Flutter 中文文档 - Flutter 中文开发者网站 - Flutter 2. 安装成功后&#xff0c;通过flutter doctor检测环境。以mac为例&#xff0c;出现了CocoaPods installed but not working 错误提示时&#xff0c;以下为解决方案: 2.1 rvm i…

AJAX: 对话框大全

AJAX:$.ajax({url: "/admin/cutting/getDataWeek",type: "GET",data:{},dataType:json,success: function (res) {if (res.code 1) {}},error:function (error) {console.log(请求失败);console.log(error);}}); $(.sub).unbind(click).click(funct…

浅析云数据安全的必要性

随着企业和个人用户越来越多地将数据存储和处理移到云上&#xff0c;云数据安全变得至关重要。云计算的发展为我们提供了卓越的灵活性和可扩展性&#xff0c;但也伴随着潜在的风险。在这个信息高度互联的时代&#xff0c;保护敏感数据是一项迫切的任务。本文将探讨云数据安全的…

microcom串口调试工具使用

microcom串口助手使用介绍 microcom是一个在终端中使用的串口助手&#xff0c;类似平常使用SSCOM一样的东西&#xff0c;不过是在终端中使用而已。 使用的是busybox构建的文件系统 microcom源码路径&#xff1a;busybox/miscutils/microcom.c microcom 参数&#xff1a; [r…

QT计时器

widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTimerEvent> //计时器类 #include <QTime> //时间类 QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_NAMESPACEclass Widget : public QWidget {Q_OBJECTpublic:Widg…

【视觉算法系列3】在自定义数据集上训练 YOLO NAS(下篇)

提示&#xff1a;免费获取本文涉及的完整代码与数据集&#xff0c;请添加微信peaeci122 YOLO-NAS是目前最新的YOLO目标检测模型&#xff0c;它在准确性方面击败了所有其他 YOLO 模型。与之前的 YOLO 模型相比&#xff0c;预训练的 YOLO-NAS 模型能够以更高的准确度检测更多目标…

C# Winform编程(7)文件处理技术

文件处理技术 System.IO命名空间File类的常用方法FileInfo类的常用方法文件夹类Directory的常用方法 System.IO命名空间 System.IO命名空间常用的类 类说明File提供用于创建&#xff0c;复制&#xff0c;删除&#xff0c;移动和打开文件的静态方法&#xff0c;并协助创建File…

python triangle库将一组闭合点转化为三角网格时网格过密的问题

输入点的格式&#xff1a; [[x1,y1], [x2,y2], … [xn,yn], ] segments 格式&#xff1a; 指示输入点的连接关系 三角化代码&#xff1a; t2 triangle.triangulate({vertices: path,segments: segments}, peq32.5a0.5)效果&#xff1a; 网格过密&#xff0c;根据文档&…

SystemVerilog Assertions应用指南 Chapter1.38在序列匹配时调用子程序

SVA可以在序列每次成功匹配时调用子程序。同一序列中定义的局部变量可以作为参数传给这些子程序。对于序列的每次匹配,子程序调用的执行与它们在序列定义中的顺序相同。 module sub;logic a, b, clk;initial $vcdpluson();initial begin clk 1b0; a1b0; b1b0; repeat(2) (pos…

代码随想录算法训练营第二十八天 | LeetCode 491. 递增子序列、46. 全排列、47. 全排列 II

代码随想录算法训练营第二十八天 | LeetCode 491. 递增子序列、46. 全排列、47. 全排列 II 文章链接&#xff1a;递增子序列 全排列 全排列II 视频链接&#xff1a;递增子序列 全排列 全排列II 目录 代码随想录算法训练营第二十八天 | LeetCode 4…

使用VGG框架实现从二分类到多分类

一.数据集的准备 与之前的不同&#xff0c;这一次我们不使用开源数据集&#xff0c;而是自己来制作数据集。重点需要解决的问题是对数据进行预处理&#xff0c;如每一个图片的大小均不同&#xff0c;需要进行resize&#xff0c;还需要对每一张图片打标签等操作。 数据集文件 …

根据pid查看jar包(windows)

打开jdk/bin/jvisualvm.exe,根据pid找到jar包的主启动类,jdk14以后不再默认使用,官网下载,也可以使用老版本的查看

虚拟机如何联网【NAT】

查看VMWARE的IP地址 #进入root用户 su -#更改虚拟网卡设置界面 vi /etc/sysconfig/network-scripts/ifcfg-ens33 修改ONBOOT为yes BOOTPROTO为static IPADDR为前面的网段 192.168.211.xx (xx为自己设置的&#xff0c;可以随意设置&#xff0c;前面的为前面查看的IP地址的前…

黑客技术(自学方法)——网络安全

前言 前几天发布了一篇 网络安全&#xff08;黑客&#xff09;自学 没想到收到了许多人的私信想要学习网安黑客技术&#xff01;却不知道从哪里开始学起&#xff01;怎么学&#xff1f;如何学&#xff1f; 今天给大家分享一下&#xff0c;很多人上来就说想学习黑客&#xff0c…

jupternotebook和jupterLab有什么区别?

目录 1.jupternotebook 2.jupterLab 3.总结 Jupyter Notebook和JupyterLab是两个常用的交互式计算环境&#xff0c;都是基于Jupyter项目开发的。它们具有一些共同的特性&#xff0c;但也存在一些区别。 1.jupternotebook Jupyter Notebook是Jupyter项目的早期版本&#xff…

【数据结构】线性表的顺序存储结构

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:数据结构 ⚙️操作环境:Visual Studio 2022 一.顺序存储定义 上篇文章中介绍了线性表一共分为两种数据结构——顺序存储结构和链式存储结构. 今天我们就来一起学习一下第一种——顺序存储结构. 线性表的顺序存储结构,指…

AUTOSAR AP硬核知识点梳理(1)

一 什么是 Adaptive AUTOSAR? Adaptive AUTOSAR是一种新的汽车软件框架,旨在满足现代汽车行业中不断增长的技术需求。随着汽车变得越来越智能,对处理器的性能要求也在不断增长。 Adaptive AUTOSAR旨在通过提供高性能计算和通信机制以及灵活的软件配置来满足这些需求,为车…