详解循环队列——链表与数组双版本

        前言:本节内容主要是讲解循环队列。 在本篇中会讲到两个版本——数组版本、链表版本。本篇内容适合正在学习数据结构队列章节或者已经学过队列但对循环队列感觉模糊的友友们 。

首先先来看一下什么是循环队列

什么是循环队列

        因为是刚开始讲解, 所以我们要先来看什么是循环队列:循环队列就是首和尾相连接的队列。如图分别是链表和数组形式的循环队列:

        循环队列普通队列相同点是:都是从队尾进数据, 从队头出数据。

        循环队列普通队列不同点是: 普通队列的容量是动态的, 会根据数据的增加而增加。 但是循环队列的容量是静态的, 它不会随着数据的增加而增加。 当队满时, 我们如果想要再添加数据, 只有将对头的数据取出来才能再次添加数据。 循环队列也被成为: 环形缓冲器

数组版本

判断队空与队满

        博主认为对于一个数组版本的循环队列来说, 最重要的就是如何判断它的队空与队满。我们先来看一下如何判断队空和队空

        

        我们利用上图进行分析。 图中tail 和 head为两个指针。 队列中的数字不是存放的数据, 而是数组的下标索引。

        我们假设初始化队空的时候tail 和 head指向同一块空间。 那么因为此时队空,所以tail 和head指向的这块空间内没有数据。

        然后我们进行入队操作, 入队一个‘5’, 这个‘5’我用绿色用来表示存放的数据。如下图:

        既然, ‘5’入队, 那么tail一定要向后移动一位, 所以结果就是上图的情况。 接下来我们再进行入队操作, 依次入队‘6’、‘1’、‘8‘、’6‘、’1‘。这些元素入队后的情况如下图: 

        ok, 到了这一步请思考一下, 这个循环队列应该是此时为满, 还是应该再入队一个元素才满? 

        我们在这里进行假设如果再入队一个元素为满。 那么再入队一个元素后,假设这个元素为‘5’,循环队列的情况就是这个样子:

现在, 请将这两个图对比着看:

        要知道, 如果此时的状态为满, 那么判满的条件就是 tail == head ; 而判空的条件我们上面说到了也是 tail == head 。那么我们如何区分这个条件下到底是满还是空?所以我们的假设是错误的。真正为满的时候应该是这样的:

         

        这个时侯其实就是该循环队列队满的情况。 那么为什么会空出来一个位置, 这个位置怎么处理? 对于这个问题。 回答是循环队列的元素个数要比开的空间数小一。 当然有别的办法解决这个问题, 但是传统的数组循环队列, 这里就是这样处理的元素个数的最大容量要比开的空间数小一

入队和出队

        第二个重要的需要搞明白的就是对于数组循环队列来说的插入和删除。指针如何偏移的问题。 

        我们在上面画的这个圆是我们想象出来的逻辑结构, 而真正的数组应该是一串连续的空间。 如下图:

        那么如果我们给这个队列入数据, 当队满时真正的物理结构其实是这种情况:

        逻辑结构是这样的情况:

       

        然后我们出一次数据:

        这是物理结构:

            这是逻辑结构:

     

        从逻辑结构我们可以看出来循环队列这个时候已经不是队满了。 但是在真正的物理结构里我们如果入数据, 那么tail指针就会越界。 如何解决这个问题呢?

        这里用到的是一点数学的运算。 

        我们设这个循环队列的最大空间是数是:maxsize。那么我们只需要在每次入数据和出数据的时候让tail或者head模上一个maxsize就可以解决这个问题。 

        比如图中head == 6, maxsize == 7. 那么当head向后移动一位时, 再取模变成 (head + 1) % maxsize

        所以, 综上,当tail 指针或者 head指针在进行入队或者出队时, 要进行的操作是 : (tail + 1) % maxsize 或者 (head + 1) % maxsize

取对头和取队尾

        取对头比较简单, 因为head指针指向的位置就是对头的位置。如图:

这个时候我们直接取对头的数据 :

data[head];  //伪代码

但是取队尾就可能有问题。 就像上图,此时tail指向了索引为0的位置。 然后在逻辑结构上面我们看到只要 tail - 1 就可以拿到索引为6的队尾数据5。但是要知道, 上图的是逻辑结构。 这个循环队列真正的物理结构应该是一块连续的数组, 就像下图这样:

  

        这个时候我们直接取 data[tail - 1]就会越界访问,显然是不正确的。

        那么解决这个问题也是用取模的方法, 但是在取模的时候要先加上一个maxsize。也就是这样

int index = (tail - 1 + maxsize) % maxsize;
data[index];    //伪代码

        当tail - 1 >= 0的时候,加上maxsize 再模上maxsize相当于将加上的maxsize又消去了。 

        当tail - 1 < 0的时候, 加上maxsize就变成了小于maxsize的正数, 取模后还是它本身。 

 以上, 就是取对头和取队尾的需要注意的事项。

代码

        知道了上面的知识点后, 我们就可以着手用代码实现我们的循环队列了。 由于知识点已经讲过了, 所以代码直接贴图了。部分内容会有注释, 但不做讲解。


//重定义数据类型。 便于维护
typedef int SQDataType;

typedef struct SeqQueue 
{
	SQDataType* _data;
	int _tail;
	int _head;
	int _maxsize;
}SQ; 


//初始化                     maxsize为循环队列的最大容量
void Init_SQ(SQ* sq, size_t maxsize) 
{
	sq->_maxsize = maxsize;
	sq->_data = (SQDataType*)malloc(sizeof(SQDataType) * maxsize);
	//
	sq->_tail = 0;
	sq->_head = 0;
}

//判断队满 
bool judge_full(SQ* sq) 
{
	return (sq->_tail + 1) % sq->_maxsize == sq->_head;    //逻辑上tail + 1 == head为队满
}

//判断队空
bool judge_empty(SQ* sq) 
{
	return sq->_tail == sq->_head;                         //当tail == head队空
}

//入队列
void Push_SQ(SQ* sq, SQDataType data)                      
{
	//先判断队列是否为满, 如果满了就返回
	if (judge_full(sq))
	{
		printf("栈满\n");
		return;

	}

	sq->_data[sq->_tail] = data;
	sq->_tail = (sq->_tail + 1) % sq->_maxsize;               //入队列要取模
}

//出队列
void Pop_SQ(SQ* sq) 
{
	//判断队列是否为空, 如果为空就返回
	if (judge_empty(sq))                                     
	{
		printf("栈空\n");
		return;
	}

	sq->_head = (sq->_head + 1) % sq->_maxsize;
}

//取对头数据
SQDataType Top_SQ(SQ* sq) 
{
	//判断是否为空
	if (judge_empty(sq))
	{
		printf("栈空\n");
		exit(-1);
	}

	return sq->_data[sq->_head];
}

//取队尾数据
SQDataType Back_SQ(SQ* sq) 
{
	//判断是否为满
	if (judge_empty(sq))
	{
		printf("栈空\n");
		exit(-1);
	}

	return sq->_data[sq->_tail - 1];
}

链表版本

入出队以及取队中数据

        其实循环队列链表版本要比数组版本更加麻烦。 首先我在这里先抛出几个问题:

        首先, 我们让这个循环队列的初始位置仍旧是tail == head。 那么它的队满位置由上面的学习我们知道是 tail->next == head;(下图中绿色数字表示节点中存放的数据)

           

这里有两个问题: 

  • 我们如何取到队尾的元素?
  • 我们如何进行出队和入队操作?出队需要释放节点吗? 如果释放节点, 那么入队时是不是需要申请节点?又或者我们直接偏移指针就可以? 

对于这两个问题, 我们先来思考一下第一个问题:

        想要取到队尾元素, 就要拿到tail指向节点的前一个节点。 那么就有两个办法解决这个问题——第一个办法就是创建一个前置指针指向tail的前一个节点, 如图:

        第二个办法就是弄成双向链表

这样取队尾就可以直接访问tail的前一个节点。 

        然后再思考第二个问题:我们在入队和出队的时候需要释放节点和申请节点吗?

        那么请看如果我们释放节点的时候会发生什么情况?如图是该循环队列的某个状态:

在这个状态下, 我们如果出队, 那么释放节点后让head指向下一个节点:

那么注意, 现在的容量减少了。 我们如果再进行入队, 假设我们入一个’5‘。 那么就变成了

好, 现在我们再对比一下这个状态和开始的状态:

        这两种状态, 是不是就是相当于head指针从一开始的指向1那个节点, 然后向后偏移一个节点。 而tail指针也相当于向后偏移了一个节点。 那么我们还有必要释放和申请节点来进行操作吗? 是不是只要偏移指针就可以了?这样是不是减少了申请和释放节点的成本, 更快更简便?

        所以, 综上, 我们就可以推断出, 链表的入队和出队操作我们不必要申请和释放节点, 和数组版本一样, 只要指针偏移即可 

循环队列的初始化

        链表循环队列另一个比较重要的要搞清楚的就是 这个队列的初始化结构是什么样的?

        其实,我们在初始化的时候将所有节点开好就可以, 然后让tail指针和head指针指向同一个节点。如图:

代码

        知道上面的所有注意事项后, 我们就可以设计链表循环队列了。 这里使用前置prevtail解决取队尾的问题。 如下为代码:

typedef int QDataType;

typedef struct QueueNode
{
	QDataType _data;
	QueueNode* _next;
}QNode;

struct Queue 
{
	QNode* _head;
	QNode* _tail;
	QNode* _prevtail;
};


//链表队列初始化
void Init_Q(Queue* pq, int n) 
{
	pq->_head = NULL;
	pq->_tail = NULL;
	QNode* cur = NULL;
	while (n) 
	{
		if (cur == NULL)  //创建第一个节点, head, tail , prev都指向第一个节点 
		{
			cur = pq->_head = pq->_tail = pq->_prevtail = new QNode;
			cur->_next = NULL;
		}
		else              //创建之后的节点
		{
			cur->_next = new QNode;
			cur = cur->_next;
		}
		n--;
	}
	cur->_next = pq->_head; //成环
}

//判断空
bool Empty_Q(Queue* pq)
{
	return pq->_head == pq->_tail;
}

//判断队满
bool Full_Q(Queue* pq)
{
	return pq->_tail->_next == pq->_head;
}

//入队列
void Push_Q(Queue* pq, QDataType x) 
{
	if (Full_Q(pq)) 
	{
		printf("栈满\n");
		return;

	}
	//将数据给tail,prev到tail的位置,tail向后偏移一位
	pq->_tail->_data = x;
	pq->_prevtail = pq->_tail;
	pq->_tail = pq->_tail->_next;
}

//出队列
void Pop_Q(Queue* pq) 
{
	if (Empty_Q(pq))
	{
		printf("栈空\n");
		return;

	}

	//head指针向后移动一位即可
	pq->_head = pq->_head->_next;
}


//取队头
QDataType Front_Q(Queue* pq) 
{
	if (Empty_Q(pq))
	{
		printf("栈空\n");
		return -1;

	}

	return pq->_head->_data;
}

//取队尾
QDataType Back_Q(Queue* pq) 
{

	if (Empty_Q(pq))
	{
		printf("栈空\n");
		return -1;

	}
	return pq->_prevtail->_data;
}

--------------------------------------------------------------------------------------------------------------------------------

以上就是本节的全部内容。

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

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

相关文章

【基础绘图】 10.饼图

效果图&#xff1a; 主要步骤&#xff1a; 1. 数据准备&#xff1a;自己赋值的随机数 2. 图像绘制&#xff1a;绘制饼图 详细代码&#xff1a;着急的直接拖到最后有完整代码 步骤一&#xff1a;导入库包及图片存储路径并设置中文字体为宋体&#xff0c;西文为新罗马&#…

totoriseSVN 常见问题

1. SVN 无法 clean up 上传时没有关闭 Excel&#xff0c;导致传入了一些临时文件&#xff08;文件名以$开头&#xff09;&#xff0c;关闭文件后临时文件自动删除&#xff0c;导致 SVN 版本错乱&#xff0c;使用 CleanUp 功能无效 更新时提示【Previous operation has not fin…

win7 phpstudy 多站点无法保存hosts的原因

1、先找到hosts文件位置 C:\Windows\System32\drivers\etc hosts文件不是txt的后缀&#xff0c;它是一个系统文件 2、如果不显示需要查找隐藏文件 组织-》文件夹和搜索选项-》查看-》取消隐藏文件夹的的√ 3、文件无法编辑 属性不要勾选只读

【SAP-FICO】SAP-FICO生产订单-结算规则配置路径(OKO7)

需求&#xff1a; 作为一个ABAPer&#xff0c;有接到一个狗屁倒灶的配置需求&#xff0c;要求如下&#xff0c;给生产订单的结算规则显示出来 图1&#xff1a;找一个生产订单&#xff0c;显示其结算规则 CO03→菜单栏-表头→结算规则 图2&#xff1a;查看该生产订单&#xff0c…

SMB/RPC协议分析之-命名/匿名管道pipe

在前面的文章中&#xff0c;介绍了SMB协议共享相关的内容&#xff0c;详见我的专栏《网络攻防协议实战分析》&#xff0c;连接这里。在SMB协议中往往需要连接到对应的远程管道&#xff0c;如果你经常接触到SMB协议&#xff0c;相信你对于lsass&#xff0c;svcctl等多种命名管道…

数据结构-二叉树-AVL树(平衡二叉树)

红黑树是平衡二叉树的一个变种。 一、 产生平衡二叉树的原因。 二叉搜索树的问题在于极端场景下退化为类似链表的结构&#xff0c;所以搜索的时间复杂度就变成了O(N)。为了保证二叉树不退化为链表&#xff0c;我们必须保证二叉树的的平衡性。 二叉平衡搜索树就是解决上面的问…

职场新人小王的沟通挑战与成长

近日&#xff0c;职场新人小王遇到了一个沟通上的小难题。作为刚刚踏入社会的新鲜人&#xff0c;小王在工作会议上因为一次直接的反馈而无意间触动了同事的敏感神经&#xff0c;导致双方关系稍显紧张。 在一次团队会议上&#xff0c;小王被要求分享对项目进度的看法以及建议。他…

【图解计算机网络】TCP 重传、滑动窗口、流量控制、拥塞控制

TCP 重传、滑动窗口、流量控制、拥塞控制 TCP 重传超时重传快速重传 滑动窗口流量控制拥塞控制慢启动拥塞避免拥塞发生快速恢复 TCP 重传 TCP重传是当发送的报文发生丢失的时候&#xff0c;重新发送丢失报文的一种机制&#xff0c;它是保证TCP协议可靠性的一种机制。 TCP重传…

9. SVG中的text元素

SVG (Scalable Vector Graphics) 提供了强大的文本渲染能力&#xff0c;其中<text>元素是常用 的文本操作的元素。本文将详细介绍<text>标签的基本使用方法&#xff0c;并展示如何通过<tspan>和<textPath>增强文本的表现力。 <text>标签基础 &…

【PHP【实战项目】系统性教学】——使用最精简的代码完成用户的登录与退出

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;开发者-曼亿点 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 曼亿点 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a…

MyBatis——MyBatis 参数处理

一、单个简单类型参数 简单类型包括&#xff1a; byte short int long float double char Byte Short Integer Long Float Double Character String java.util.Date java.sql.Date parameterType 属性&#xff1a;告诉 MyBatis 参数的类型 MyBatis 自带类型自动推断机制…

【Linux】centos7安装软件(rpm、yum、编译安装),补充:查找命令的相关文件路径,yum安装mysql

【Linux】技术上&#xff0c;Linux是内核。而术语上&#xff0c;我们通常说的Linux是完整的操作系统&#xff0c;其实称为"Linux发行版"&#xff0c;是将Linux内核和应用系统打包&#xff0c;由不同的发行家族发行了不同版本。Linux发行版众多&#xff0c;主要有RedH…

Debian常用命令:高效管理与运维的必备指南

在Linux世界中&#xff0c;Debian以其稳定性、安全性和开源精神赢得了广大用户的青睐。作为一个基于Linux的操作系统&#xff0c;Debian拥有丰富且强大的命令行工具&#xff0c;这些命令对于系统管理员和开发者来说至关重要。本文将为您介绍一系列Debian系统中的常用命令&#…

基于Javaee的影视创作论坛的设计与实现+vue论文

系统简介 传统办法管理信息首先需要花费的时间比较多&#xff0c;其次数据出错率比较高&#xff0c;而且对错误的数据进行更改也比较困难&#xff0c;最后&#xff0c;检索数据费事费力。因此&#xff0c;在计算机上安装影视创作论坛软件来发挥其高效地信息处理的作用&#xf…

询问贴:这要怎么设置捏,寻思着总不该一个一个挖空吧????

这要怎么设置捏&#xff0c;寻思着总不该一个一个挖空吧&#xff1f;&#xff1f;&#xff1f;&#xff1f;

【javaSE】认识异常(1)

❤️❤️前言~&#x1f973;&#x1f389;&#x1f389;&#x1f389; hellohello~&#xff0c;大家好&#x1f495;&#x1f495;&#xff0c;这里是E绵绵呀✋✋ &#xff0c;如果觉得这篇文章还不错的话还请点赞❤️❤️收藏&#x1f49e; &#x1f49e; 关注&#x1f4a5;&a…

联丰策略股票杠杆股票交易市场突破3100点!A股稳了?

查查配近期,大盘再次来到3100点附近。 重要关口得到有效突破,市场情绪明显升温,甚至有投资者高喊:反转已经在路上!但也有谨慎者认为,市场仍有回调风险,围绕3000点震荡或是接下来的主旋律。 联丰策略拥有一支由知名互联网公司和国内证券金融机构的行业专家组成的一流运营团队。…

HTML炫酷的相册

目录 写在前面 HTML简介 完整代码 代码分析 系列推荐 写在最后 写在前面 本期小编给大家带来一个炫酷的旋转相册&#xff0c;快来解锁属于你的独家记忆吧&#xff01; HTML简介 HTML&#xff08;全称为超文本标记语言&#xff09;是一种用于创建网页结构和内容的标记语…

Python 编程语言中的 None 到底是什么?

&#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 让我们一起深入了解 Python 中的 None。 什么是 None&#xff1f; 在 Python 编程语言中&#xff0c;None 是一个特殊的常量&#xff0c;它代表了 “无” 或 “没有值”。你可以把它想象成一个空盒子…

日本率先研发成功6G设备,刺痛了谁?为何日本能率先突破?

日本率先研发成功6G设备&#xff0c;无线数据速率是5G的百倍&#xff0c;这让日本方面兴奋莫名&#xff0c;毕竟日本在科技方面从1990年代以来太缺少突破的创新了&#xff0c;那么日本为何如今在6G技术上能率先突破呢&#xff1f; 日本在1980年代末期达到顶峰&#xff0c;它的科…