循环队列与循环双端队列

news2024/11/15 8:35:46

文章目录

  • 前言
  • 循环队列
  • 循环双端队列


前言

1、学习循环队列和循环双端队列能加深我们对队列的理解,提高我们的编程能力。
2、本文循环队列使用的是数组,循环双端队列用的是双向链表
3、题目连接:设计循环队列 ,设计循环双端队列。

循环队列

1、什么是循环队列?

循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环。它也被称为“环形缓冲器”。

在这里插入图片描述

2、实现的功能

(1)MyCircularQueue(k): 构造器,设置队列长度为 k 。
(2)Front: 从队首获取元素。如果队列为空,返回 -1。
(3)Rear: 获取队尾元素。如果队列为空,返回 -1 。
(4)enQueue(value):向循环队列插入一个元素。如果成功插入则返回真。
(5)deQueue(): 从循环队列中删除一个元素。如果成功删除则返回真。
(6)isEmpty(): 检查循环队列是否为空。
(7)isFull(): 检查循环队列是否已满。

3、设计
有两种方案:a:利用数组来存储数据,b:利用链表来存储数据
我们这里使用数组的方式
(1)我们设置一个头指针,和一个尾指针(指的时最后一个有数据位置的下一个位置,为什么不直接指向最后有数据那个位置呢?因为这样能更好的判断队列是否为空和队列是否已经满的情况),头指针(front),尾指针(rear),容量(k)。

(2) 为了解决尾指针指向最后一个数据后一个的问题,我们可以多申请一个空间,就不会使尾指针指的位置超出数组了,这个问题也叫假溢出。

(3)判断空,当front=rear时为空
![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/d05483fe9eb840fdad065ce02b088b52.png

(4)判断满,当 front=(rear+1)%(k+1) 为空

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/d06eadb22f7048aa8ede33702d1e9548.png

(5)删除队头元素,使front=(front+1)%(k+1)即可, 也可以通过判断front是否在(k+1)的位置,在的话就使front=0,不在的话front=front+1即可

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/14aff520fd254a0a9b2751cf622e4a82.png

(6)进队,将数据放到数组下标rear位置上,然后使 rear=(rear+1)%(k+1) 即可,原理同上。

(7)获取队头元素,直接返回队头下标的位置的数据即可

(8)获取队尾元素,返回 (rear-1+k+1)%(k+1) 位置的数据即可,也可以判断rear是不是在0的位置,在的话top=k,不在0的位置的话就top=rear-1
![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/fad58d936d8b45a0b6b65024f18fd026.png

4、代码实现:

//队列结构体
typedef struct {
    //储存数据
    int *arr;
    //头
    int front;
    //指向尾的下一个
    int rear;
    //大小
    int k;
} MyCircularQueue;

//初始化循环队列
MyCircularQueue* myCircularQueueCreate(int k) {
    //队列
     MyCircularQueue* obj=( MyCircularQueue*)malloc(sizeof(MyCircularQueue));

     //初始化成员
     obj->arr=(int*)malloc(sizeof(int)*(k+1));
     obj->front=0;
     obj->rear=0;
     obj->k=k;
     
     return obj;
}

//是否为空
bool myCircularQueueIsEmpty(MyCircularQueue* obj) {
    //当队头的下标等于队尾的下标时队列为空
    return obj->front==obj->rear;
}

//是否为满
bool myCircularQueueIsFull(MyCircularQueue* obj) {
    //当队头的下标等于队尾加一模上k+1时队列满了
    return obj->front==(obj->rear+1)%(obj->k+1);
}

//入队
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {
    //当队列满了就返回false
        if(myCircularQueueIsFull(obj))
           return false;

    //放到rear位置上
        obj->arr[obj->rear]=value;
    //这样当rear+1==k+1时,让rear回到0这个位置上
        obj->rear=(obj->rear+1)%(obj->k+1);

           return true;
}

//出队
bool myCircularQueueDeQueue(MyCircularQueue* obj) {
    //空时返回false
    if(myCircularQueueIsEmpty(obj))
       return false;
      //队头下标加1,莫上k+1当front+1==k+1时能回到0那个位置
     obj->front=(obj->front+1)%(obj->k+1);

     return true;
}

//查看头元素
int myCircularQueueFront(MyCircularQueue* obj) {
     //空时返回-1
    if(myCircularQueueIsEmpty(obj))
      return -1;

     //直接展示front位置即可
      int tmp=obj->arr[obj->front];

      return tmp;
}

//查看队尾元素
int myCircularQueueRear(MyCircularQueue* obj) {
    //空时返回-1
    if(myCircularQueueIsEmpty(obj))
      return -1;
    //因为返回的是rear-1位置上的数据,当rear>0时,查看的位置时rear-1,当rear=0时就是查看k位置的数据了
      int tmp=obj->arr[(obj->rear-1+obj->k+1)%(obj->k+1)];

      return tmp;
    
}

//释放
void myCircularQueueFree(MyCircularQueue* obj) {
    free(obj->arr);
    free(obj);
}

循环双端队列

1、循环双端队列就是在循环队列的基础上增加了一些接口,如:可以进行队头的插入,进行尾部的删除。

2、实现的功能接口:

实现 MyCircularDeque 类:

(1)MyCircularDeque(int k) :构造函数,双端队列最大为 k 。

(2)boolean insertFront():将一个元素添加到双端队列头部。 如果操作成功返回 true ,否则返回 false 。

(3)boolean insertLast() :将一个元素添加到双端队列尾部。如果操作成功返回 true ,否则返回 false 。

(4)boolean deleteFront() :从双端队列头部删除一个元素。 如果操作成功返回 true ,否则返回 false 。

(5)boolean deleteLast() :从双端队列尾部删除一个元素。如果操作成功返回 true ,否则返回 false 。

(6)int getFront() ):从双端队列头部获得一个元素。如果双端队列为空,返回 -1 。

(7)int getRear() :获得双端队列的最后一个元素。 如果双端队列为空,返回 -1 。
(8) boolean isEmpty() :若双端队列为空,则返回 true ,否则返回 false 。

(9)boolean isFull() :若双端队列满了,则返回 true ,否则返回 false 。

3、设计
(1)用双向链表的节点,这样方便找到尾部的上一个节点,利于队尾的删除。
(2)定义size来判断空和满,再定义两个节点指针,分别指向队头(front)队尾(rear)容量(k)前驱指针(next)后驱指针(next1)
(3)判断为空,当size=0时为空。
(4)判断是否满,当size=k时为满。
(5)队头插入数据,申请一个节点tmp,再将tmp和front连起来,最后front=tmp即可

在这里插入图片描述

(6)队尾插入数据,申请一个节点tmp,再将tmp和rear连接起来,最后rear=tmp即可
在这里插入图片描述

(7)队头删除,先保存队头的后一个节点next,再将front释放,最后将front=next并把front->next1=NULL即可,注意顺序不能乱。

在这里插入图片描述

(8)队尾删除,先保存前一个节点next1,再将rear释放,最后将rear=next1并把rear->next=NULL即可,注意顺序不能乱。

在这里插入图片描述

(9)获取队头元素,直接返回front的数据即可。
(10)获取队尾元素,直接返回rear的数据即可。

4、代码实现:

//双向链表的结构体
typedef struct  ls
{
     //前驱
   struct ls *next;
   //后驱
   struct ls *next1;
   //数据
   int val;
}ls;

class MyCircularDeque {
public:
     //初始化
    MyCircularDeque(int k) {
    //容量
         this->k=k;
      //大小
         this->size=0;
         this->rear=NULL;
         this->front=NULL;
    }
    
    //队头插入数据
    bool insertFront(int value) {
    //  满了,就返回
           if(isFull())
             return false;

      //没满
      //申请一个节点
             ls *tmp(ls*)malloc(sizeof(ls));
             tmp->next=NULL;
             tmp->next1=NULL;
             tmp->val=value;

       //空的话头和尾指向第一个节点
             if(front==NULL)
             {
               front=rear=tmp;
             }
             //不空,插入头
             else
             {
                 front->next1=tmp;
                 tmp->next=front;
                 front=tmp;
             }
             
          //大小加1
             size++;

            return true;
    }
    
    //队尾插入数据
    bool insertLast(int value) {
              if(isFull())
             return false;

      //申请节点
             ls *tmp=(ls*)malloc(sizeof(ls));
             tmp->next=NULL;
             tmp->next1=NULL;
             tmp->val=value;
             
             if(rear==NULL)
             {
                 front=rear=tmp;
             }
             //尾插到rear后面
             else
             {
                   tmp->next1=rear;
                   rear->next=tmp;
                   rear=tmp;
             }
             
     //大小加1
             size++;

             return true;
    }
    
 //删队头   
    bool deleteFront() {
    //为空
        if(isEmpty())
         return false;
         
         //只有一个元素
         ls *tmp=front->next;
         free(front);
         if(tmp==NULL)
           front=rear=tmp;
           //多个元素
         else
         {
        front=tmp;
        front->next1=NULL;
         }
       
       //大小-1     
         size--;

          return true;
    }
    
    //删队尾
    bool deleteLast() {
          if(isEmpty())
         return false;
         
         //只有一个元素
         ls *tmp=rear->next1;
         free(rear);

         if(tmp==NULL)
         {
             rear=front=tmp;
         }
         else
         {
             tmp->next=NULL;
             rear=tmp;
         }
    // 大小-1
         size--;

          return true;
    }
    
    //显示头元素
    int getFront() {
    //为空
          if(isEmpty())
         return -1;

//直接返回头节点的数据
         return front->val;
    }
    
    //显示尾节点元素
    int getRear() {
    //为空
        if(isEmpty())
         return -1;
         
//直接返回尾节点数据
         return rear->val;

    }
    
    //判断是否为空
    bool isEmpty() {
    //当size==0是为空
       return size==0;
    }
    
    //判断是否为满
    bool isFull() {
    //size==k就满了
      return size==k;
    }
    
//释放链表
    ~MyCircularDeque()
    {
        ls* tmp=front;
        while(tmp)
        {
            ls*p=tmp->next;
            free(tmp);
            tmp=p;
        }
    }
    //头和尾
    ls* front;
    ls* rear;
    //容量和大小
    int k;
    int size;
};

以上就是我的分享了

最后,感谢大家的观看!

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

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

相关文章

C++面试宝典第34题:整数反序

题目 给出一个不多于5位的整数, 进行反序处理。要求: 1、求出它是几位数。 2、分别输出每一位数字。仅数字间以空格间隔, 负号与数字之间不需要间隔。如果是负数,负号加在第一个数字之前, 与数字没有空格间隔。注意:最后一个数字后没有空格。 3、按逆序输出各位数字。逆序后…

华为od机试C卷-开源项目热度榜单

1、题目描述 某个开源社区希望将最近热度比较高的开源项目出一个榜单,推荐给社区里面的开发者。 对于每个开源项目,开发者可以进行关注(watch)、收藏(star)、fork、提issue、提交合并请求(MR)等。 数据库里面统计了每个开源项目关注、收藏、fork、issue…

适用于恢复iOS数据的 10 款免费 iPhone 恢复软件

现在,您可以获得的 iPhone 的存储容量比大多数人的笔记本电脑和台式电脑的存储容量还要大。虽然能够存储数千张高分辨率照片和视频文件、安装数百个应用程序并随身携带大量音乐库以供离线收听固然很棒,但在一个地方拥有如此多的数据可能会带来毁灭性的后…

LNOI省选祭录

写在前面 大概率爆零 你说得对,但是「辽宁省2024联合省选」是一款由「中国计算机学会」推出的一款「开放世界冒险游戏」,游戏发生在一个被称作「大连大学」的幻想世界,在这里,被神选中的人将被授予「xor魔法手杖」,引…

#WEB前端(浮动与定位)

1.实验&#xff1a; 2.IDE&#xff1a;VSCODE 3.记录&#xff1a; float、position 没有应用浮动前 应用左浮动和右浮动后 应用定位 4.代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><me…

matplotlib直方图

matplotlib直方图 假设你获取了250部电影的时长(列表a中), 希望统计出这些电影时长的分布状态(比如时长为100分钟到120分钟电影的数量, 出现的频率)等信息, 你应该如何呈现这些数据? from matplotlib import pyplot as plt a[131, 98, 125, 131, 124, 139, 131, 117, 128, …

linux逻辑卷管理

一.物理卷&#xff0c;逻辑卷&#xff0c;卷组的关系 二.实验题目 1.业务需要&#xff0c;新增5G硬盘&#xff0c;先对第一块磁盘分区&#xff0c;大小为4G&#xff0c;现在进行逻辑卷划分&#xff0c;卷组名为myvg,逻辑卷名为LV1&#xff0c;大小为2G 2.格式化逻辑卷LV1&#…

【论文阅读】多传感器SLAM数据集

一、M2DGR 该数据集主要针对的是地面机器人&#xff0c;文章正文提到&#xff0c;现在许多机器人在进行定位时&#xff0c;其视角以及移动速度与车或者无人机有着较大的差异&#xff0c;这一差异导致在地面机器人完成SLAM任务时并不能直接套用类似的数据集。针对这一问题该团队…

附加Numpy数组

参考&#xff1a;Append Numpy Array 引言 在数据科学和机器学习领域&#xff0c;处理大规模数据集是一项重要且常见的任务。为了高效地处理数据&#xff0c;numpy是一个非常强大的Python库。本文将详细介绍numpy中的一个重要操作&#xff0c;即如何附加&#xff08;append&a…

【网站项目】308学生档案管理系统

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

【大数据Hive】hive 多字段分隔符使用详解

目录 一、前言 二、hive默认分隔符规则以及限制 2.1 正常示例&#xff1a;单字节分隔符数据加载示例 2.2 特殊格式的文本数据&#xff0c;分隔符为特殊字符 2.2.1 文本数据的字段中包含了分隔符 三、突破默认限制规则约束 3.1 数据加载不匹配情况 1 3.2 数据加载不匹配…

【Redis 主从复制】

文章目录 1 :peach:环境配置:peach:1.1 :apple:三种配置方式:apple:1.2 :apple:验证:apple:1.3 :apple:断开复制和切主:apple:1.4 :apple:安全性:apple:1.5 :apple:只读:apple:1.6 :apple:传输延迟:apple: 2 :peach:拓扑结构:peach:2.1 :apple:⼀主⼀从结构:apple:2.2 :apple:⼀…

最简单的基于 FFmpeg 的收流器(以接收 RTMP 为例)

最简单的基于 FFmpeg 的收流器&#xff08;以接收 RTMP 为例&#xff09; 最简单的基于 FFmpeg 的收流器&#xff08;以接收 RTMP 为例&#xff09;正文结果工程文件下载参考链接 最简单的基于 FFmpeg 的收流器&#xff08;以接收 RTMP 为例&#xff09; 参考雷霄骅博士的文章…

Easy Graphics Engine on GitHub

Easy Graphics Engine 组织 Easy-Graphics-Engine 此组织的创建是为了方便以后分享和维护项目。目前仅整理了少量几个项目&#xff0c;后面会不断进行扩充。 GitHub EGE 项目列表 ege-project-list 分类整理 GitHub 上使用 EGE 开发的项目&#xff0c;便于用户学习。后续会不断…

操作系统(1)——学习导论(Ⅱ)

目录 小程一言专栏链接: [link](http://t.csdnimg.cn/6grrU) 学习导论&#xff08;Ⅱ&#xff09;操作系统-赏前人佳作大型操作系统大型操作系统的一些特点和功能举例 服务器操作系统服务器操作系统特点和功能举例 多处理器操作系统举例 个人计算机操作系统举例 掌上计算机操作…

二十三、剖析 LinkedList

剖析 LinkedList 本文为书籍《Java编程的逻辑》1和《剑指Java&#xff1a;核心原理与应用实践》2阅读笔记 ArrayList随机访问效率很高&#xff0c;但插入和删除性能比较低&#xff1b;LinkedList同样实现了List接口&#xff0c;它的特点与ArrayList几乎正好相反。除了实现了L…

【CSP试题回顾】201709-1-打酱油

CSP-201709-1-打酱油 完整代码 #include<iostream> using namespace std; int main() {int N, num 0;cin >> N;int n1 N / 50;if (n1 ! 0){N N - n1 * 50;num n1 * 7;}int n2 N / 30;if (n2 ! 0){N N - n2 * 30;num n2 * 4;}int n3 N / 10;num n3;cout…

幂等性设计

目录 前言 幂等性设计 幂等性设计处理流程 HTTP 幂等性 消息队列幂等性 机遇kafka 前言 幂等性设计&#xff0c;就是说&#xff0c;一次和多次请求某一个资源应该具有同样的副作用。为什么我们要有幂等性操作&#xff1f;说白了&#xff0c;就两点&#xff1a;1、网络的…

品牌作为储蓄池:静态积累与长期价值

在瞬息万变的商业环境中&#xff0c;品牌不仅仅是一个简单的标识或名字&#xff0c;它更像是一个融合了静态积累与长期价值的综合储蓄池。品牌通过不断的积累&#xff0c;建立起深厚的市场基础和消费者信任&#xff0c;这些无形的资产为品牌的长期发展奠定了坚实的基础。品牌推…

【大厂AI课学习笔记NO.59】(12)过拟合与欠拟合

拟合就是调整参数和模型&#xff0c;让结果无限接近真实值的过程。 我们先来了解个概念&#xff1a; 偏差-方差窘境&#xff08;bias-variance dilemma&#xff09;是机器学习中的一个重要概念&#xff0c;它涉及到模型选择时面临的权衡问题。 偏差&#xff08;Bias&#xf…