【数据结构和算法】--队列的特殊结构-循环队列

news2024/11/23 12:58:00

目录

  • 循环队列的结构
  • 循环队列的实现
    • 循环队列的创建
    • 循环队列为空判断
    • 循环队列为满判断
    • 入队
    • 出队
    • 返回循环队列首元素
    • 返回循环队列尾元素
    • 释放循环队列

循环队列的结构

循环队列是队列的一种特殊结构,它的长度是固定的k,同样是先进先出,理论结构是首尾相连的环形循环结构。其理论结构大致如下:

在这里插入图片描述

具体结构描述可以参考LeetCode: 622. 设计循环队列的题目要求,大致如下:
设计你的循环队列实现。 循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环。它也被称为“环形缓冲器”。
循环队列的一个好处是我们可以利用这个队列之前用过的空间。在一个普通队列里,一旦一个队列满了,我们就不能插入下一个元素,即使在队列前面仍有空间。但是使用循环队列,我们能使用这些空间去存储新的值。

循环队列的实现

循环队列的实现方式同样有两种–数组,链表

  1. 数组循环队列:
    数组实现方式顾名思义就是动态开辟一个长度为k的数组,那要怎么达到循环呢?我们可以定义两个数一个指向队列头元素int front),一个指向队列尾元素的下一个int back),(此处指向队尾下一个是为了方便队列空和满的判断),这样当back走到最后一个时,我们只需要将他重新置成0便又到了队列第一个元素,如此设计理论上就达到了循环结构。
    新的问题又来了:当front == back时要怎么区分队列是空还是满?两种解决方案:
  1. 增加一个size记录有效数据的节点数size == 0队列就是空,size == k队列就是满。
  2. 多开辟一个空间k+1front == back便是空,(back+1)%(k+1) == front就是满(即在理论结构中back指向的下一个位置是front,下文会讲解)。
  1. 链表循环队列:
    链表实现的循环队列更有点循环的味(即将尾指针的next指向头,形成真正的循环),但实现起来不如数组方便。链表实现循环队列同样要定义两个指针,一个指向循环队列的头元素(phead),一个指向循环队列尾元素的下一个(ptail)。判断循环队列空和满的方法和数组相似,只不过判断条件从判断值相同改为判断址相同,第二种方法判满改为phead == ptail->next
    但用链表设计循环队列也会有新的困难:1. 获取循环队列尾元素不方便,还要遍历队列寻找;2. 定义结构体时,还要多定义一个装链表节点的结构体,这也增加了代码的难度。

所以不论是用数组还是用链表实现循环队列,都有各自的好处和问题,下面实现循环队列我所介绍的方法是数组实现法判满和判空用的是多定义一个节点法

依据上述方法,可以定义如下结构体变量:

typedef struct
{
    int* a;//数组
    int front;//队列头
    int back;//队列尾下一个位置
    int k;//循环队列可存储数据长度
} MyCircularQueue;

循环队列的创建

注意此处所给的函数返回值类型MyCircularQueue,正是上述结构体类型。故须先动态开辟一个结构体类型大小,并将frontback初始化为0,然后再利用结构体指针来开辟长度为k+1的数组最后返回结构体指针
这样动态开辟而不直接定义结构体变量(MyCircularQueue ps),是因为这是在函数中,出了函数的作用域此结构体变量就会销毁,所以需要malloc将结构体动态开辟在堆区。

MyCircularQueue* myCircularQueueCreate(int k)
{
    MyCircularQueue* ps = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));
    ps->a = (int*)malloc(sizeof(int)*(k+1));
    ps->back = ps->front = 0;
    ps->k = k;
    return ps;
}

循环队列为空判断

bool myCircularQueueIsEmpty(MyCircularQueue* obj)
{
    return obj->front == obj->back;
}

循环队列为满判断

在这里插入图片描述

循环队列为满大致可以分为以上两种情况。

  1. 情况1:
    当队列尾back来到最后一个时,此时如果back + 1的话就会超过k + 1的范围,而我们的目的是想知道在循环队列中back + 1后的位置(即下标为0的位置),所以此时我们只要将(obj->back + 1)%(obj->k + 1),此式的值便为0
  2. 情况2:
    back + 1的范围在k + 1内,此时判断back + 1即可,若(obj->back + 1)%(obj->k + 1)也不会影响值。

如此将两式合并,便得到了简化的效果。

bool myCircularQueueIsFull(MyCircularQueue* obj)
{
    return (obj->back+1)%(obj->k+1) == obj->front;
}

入队

题目描述:enQueue(value):向循环队列插入一个元素。如果成功插入则返回真。
既如此那么先判断循环队列是否已满,若满则返回false,否则入队新值,并将obj->back值更新(obj->back %= obj->k+1;

bool myCircularQueueEnQueue(MyCircularQueue* obj, int value)
{
    if(myCircularQueueIsFull(obj))
        return false;
    obj->a[obj->back] = value;
    obj->back++;
    obj->back %= obj->k+1;
    return true;
}

出队

题目描述:deQueue():从循环队列中删除一个元素。如果成功删除则返回真。
与入循环队列相似。

bool myCircularQueueDeQueue(MyCircularQueue* obj)
{
    if(myCircularQueueIsEmpty(obj))
        return false;
    obj->front++;
    obj->front %= obj->k+1;
    return true;
}

返回循环队列首元素

题目描述:Front:从队首获取元素。如果队列为空,返回 -1 。
先判断循环队列不为空,若为空返回-1,不为空返回下标为obj->front的值。

int myCircularQueueFront(MyCircularQueue* obj)
{
    if(myCircularQueueIsEmpty(obj))
        return -1;
    return obj->a[obj->front];
    
}

返回循环队列尾元素

题目描述:Rear:获取队尾元素。如果队列为空,返回 -1 。
同样要先判断循环队列是否为空,为空返回-1。此处计算obj->back上一个的下标的方法中,obj->back-1后还要加obj->k+1是为了防止obj->back指向下标为0的情况,再% (obj->k+1)是为了防止超出下标范围的情况,与判断队满相似。

int myCircularQueueRear(MyCircularQueue* obj)
{
    if(myCircularQueueIsEmpty(obj))
        return -1;
    return obj->a[(obj->back-1+obj->k+1) % (obj->k+1)];
}

释放循环队列

依次释放即可,先释放最内层的数组,然后释放最外层的结构体。

void myCircularQueueFree(MyCircularQueue* obj)
{
    free(obj->a);
    free(obj);
}

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

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

相关文章

飞天使-docker知识点6-容器dockerfile各项名词解释

文章目录 docker的小技巧dockerfile容器为什么会出现启动了不暂停查看docker 网桥相关信息 docker 数据卷 docker的小技巧 [rootlight-test playbook-vars[]# docker inspect -f "{{.NetworkSettings.IPAddress}}" d3a9ae03ae5f 172.17.0.4docker d3a9ae03ae5f:/etc…

RK3399平台开发系列讲解(内核入门篇)什么是函数调用栈

🚀返回专栏总目录 文章目录 一、什么是函数调用栈二、函数调用栈解析三、什么是stack overflow沉淀、分享、成长,让自己和他人都能有所收获!😄 📢在开发软件的过程中我们经常会遇到错误,如果你用 Google 搜过出错信息,那你多少应该都访问过Stack Overflow这个网站。作…

三、JS逆向

一、JS逆向 解释:在我们爬虫的过程中经常会遇到参数被加密的情况,这样只有先在前端搞清楚加密参数是怎么生成的才能继续我们的爬虫,而且此时我们还需要用python去执行这个加密的过程。本文主要讲怎么在浏览器调试JS,以及Python执…

网络(七)路由协议以及相关配置

目录 一、路由器的工作原理 二、路由表的形成 2.1 直连网段 2.2 非直连网 2.3 路由表解析 2.3.1 查看路由表 2.3.2 解析 三、静态路由和默认路由 1. 静态路由 1.1 定义 1.2 特点 2. 默认路由 2.1 定义 2.2 特点 四、静态路由和默认路由的配置 1. 静态路由配置…

很抱歉,Midjourney,但Leonardo AI的图像指导暂时还无人能及…至少目前是这样

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗?订阅我们的简报,深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同,从行业内部的深度分析和实用指南中受益。不要错过这个机会,成为AI领…

汽车充电协议OpenV2G的平替cbexigen!!

纵所周知,开源欧规协议 CCS 的 OpenV2G 协议不支持 ISO15118-20:2022 协议,并且软件维护者已经明确不在进行该软件的维护。 前几天在 Github 上冲浪发现了一个宝藏开源项目,完美的实现的 OpenV2G 的 Exidizer 工具的功能:cbexigen…

文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《考虑灵活性资源传输精细化建模的配电网优化运行》

这个标题表达的是关于配电网优化运行的一个概念,其中考虑了灵活性资源传输的精细化建模。让我们逐个解读关键词: 考虑灵活性资源传输:这指的是在配电网优化运行中考虑到不同类型的灵活性资源的传输。灵活性资源包括可再生能源、储能系统、柔性…

2044回文字符串(C语言)

目录 一:题目 二:思路分析 1.什么是回文? 2.判断回文: 三:代码 一:题目 二:思路分析 1.什么是回文? 最简单的理解方式就是一个字符串正着写和倒着写一样 2.判断回文&#xff1…

【C语言(十)】

字符函数和字符串函数 一、字符分类函数 C语言中有⼀系列的函数是专门做字符分类的,也就是⼀个字符是属于什么类型的字符的。这些函数的使用都需要包含⼀个头文件是 ctype.h 这些函数的使用方法非常类似,我们就讲解⼀个函数的事情,其他的非…

代码随想录-刷题第二十九天

491. 递增子序列 题目连接:491. 递增子序列 思路:将问题抽象为树形结构,使用回溯法。本题求自增子序列,是不能对原数组进行排序的,排完序的数组都是自增子序列了。所以不能使用之前的去重逻辑! HashSet是…

【人工智能】实验二: 洗衣机模糊推理系统实验与基础知识

实验二: 洗衣机模糊推理系统实验 实验目的 理解模糊逻辑推理的原理及特点,熟练应用模糊推理。 实验内容 设计洗衣机洗涤时间的模糊控制。 实验要求 已知人的操作经验为: “污泥越多,油脂越多,洗涤时间越长”;“…

DDD挤水分和强行加异性为好友-UMLChina建模知识竞赛第4赛季第25轮

DDD领域驱动设计批评文集 做强化自测题获得“软件方法建模师”称号 《软件方法》各章合集 参考潘加宇在《软件方法》和UMLChina公众号文章中发表的内容作答。在本文下留言回答。 只要最先答对前3题,即可获得本轮优胜。第4题为附加题,对错不影响优胜者…

C++ Qt开发:标准Dialog对话框组件

Qt 是一个跨平台C图形界面开发库,利用Qt可以快速开发跨平台窗体应用程序,在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置,实现图形化开发极大的方便了开发效率,本章将重点介绍标准对话框QInputDialog、QFileDialog 这两种…

阿里云RDS提示过期释放实例了怎么找到库表

做朋友,不需要资格!——《全职猎人》 直截了当 一步到位 ~

Flink系列之:窗口函数Windowing table-valued functions (Windowing TVFs)

Flink系列之:窗口函数Windowing table-valued functions Windowing TVFs 一、窗口函数二、Tumble Windows三、Hop Windows四、Cumulate Windows四、Window Offset 适用流、批 Windows 是处理无限流的核心。 Windows 将流分割成有限大小的“桶”,我们可以…

Android动画(二)——补间动画

目录 介绍 Xml文件定义View动画 补充 alpha_animation.xml(透明度) rotate_animation.xml(旋转) scale_animation.xml(伸缩) translate_animation.xml (平移) group_animation.…

STM32迪文屏图标控件保姆级教程

要主图的去末尾,末尾福利图在等着你~~~ 文章目录 前言 开发环境 二、使用步骤 1.添加图标控件 2.设置图标属性 3.图标库ICL文件生成 4.单片机程序编写 容易踩得坑 一、前言 本篇文章主要介绍了在DGBUS平台上使用图标变量的步骤。首先需要在DGBUS中添加一个图标变量控…

项目抛异常:‘Date‘ 不是可以识别的 内置函数名称。 关键字 ‘AS‘ 附近有语法错误。

今天在解决公司C#项目bug问题,顺便记录一下问题及解决方案。入库日报表和出库日报表执行查询失败: ‘DATE’ 不是可以识别的 内置函数名称。 问题截图如下: 这个错误信息看起来是由于在执行远程方法 SIE.WMS.Statistics.StatisticsControll…

Windows 10如何关闭系统自动更新(实用教程)

本章教程,用最简洁的方式介绍在windows10中如何关闭系统自动更新。 目录 一、关闭自动更新服务 二、关闭自动更新组策略 一、关闭自动更新服务 1、 winr 2、services.msc 3、找到并双击 Windows Update 修改启动类型为禁用 二、关闭自动更新组策略 1、winr 2、gp…

【算法Hot100系列】最长回文子串

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…