什么?要求设计一个循环队列?

这篇具有很好参考价值的文章主要介绍了什么?要求设计一个循环队列?。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

什么?要求设计一个循环队列?

🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨
🐻推荐专栏:
🍔🍟🌯C语言初阶
🍔🍟🌯C语言进阶

🔑个人信条: 🌵知行合一
🍉本篇简介:>:讲解用c语言实现数据结构的循环队列.

一、题目介绍:

先声明一下:
题目来源:力扣(LeetCode)

题目名称:设计循环队列:题目链接
难度: 中等

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

循环队列的一个好处是我们可以利用这个队列之前用过的空间。在一个普通队列里,一旦一个队列满了,我们就不能插入下一个元素,即使在队列前面仍有空间。但是使用循环队列,我们能使用这些空间去存储新的值。

需要实现的接口介绍:

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

测试接口示例:

MyCircularQueue circularQueue = new MyCircularQueue(3); // 设置长度为 3
circularQueue.enQueue(1); // 返回 true
circularQueue.enQueue(2); // 返回 true
circularQueue.enQueue(3); // 返回 true
circularQueue.enQueue(4); // 返回 false,队列已满
circularQueue.Rear(); // 返回 3
circularQueue.isFull(); // 返回 true
circularQueue.deQueue(); // 返回 true
circularQueue.enQueue(4); // 返回 true
circularQueue.Rear(); // 返回 4

二、接口函数的分析:

2.1 循环队列的结构

typedef int Queuedate;
typedef struct {
    Queuedate* date;
    int front;//队首下标
    int rear;//队尾
    int k;//队列的长度(固定的)
} MyCircularQueue;

刚开始设计循环队列时:
什么?要求设计一个循环队列?
为了显示循环的模样,更加形象的图:
什么?要求设计一个循环队列?

此时遇到了一个难题:

什么?要求设计一个循环队列?

什么?要求设计一个循环队列?
什么?要求设计一个循环队列?

为了解决队列判满与判空冲突问题,这里选择设计2:以多开一个空间为代价.

那有没有办法不开空间也能解决这个问题呢?

另外方案:
增加一个size指针,用于记录循环队列元素的实际元素个数.

满队列: size=k
空队列: size为0的时候是空队列

2.2 初始化"循环队列"(myCircularQueueCreate)

步骤:

  1. 为使得myCircularQueueCreate函数生命周期结束后,obj(循环队列)不被销毁,所以需要动态申请(malloc)空间.
  2. obj(循环队列)的date 指针申请k(容量)个单位的空间.
  3. front (队首下标)和rear(待插入位置下标)设置初始状态为0.
  4. 将参数k的值,存入obj(循环队列)保存,作为循环队列最大容量.
  5. 返回obj(循环队列).

代码:

MyCircularQueue* myCircularQueueCreate(int k) {
    MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));;
    obj->date = (Queuedate*)malloc(sizeof(Queuedate) * (k + 1));
    if (obj->date == NULL)//如果开辟空间失败
    {
        perror("obj malloc error");
    }
    //初始化时,队首和队尾都暂时赋值为0下标
    obj->front = obj->rear = 0;
	//记录k的值,k表示循环队列的容量.
    obj->k = k;
    return obj;
}

2.3 入队(myCircularQueueEnQueue)

返回值说明:

true表示入队成功.
false表示入队失败.

步骤:

1.进行入队操作前,需要考虑队满情况(队满直接返回false入队失败).
2.在rear下标位置插入新元素value.
3. 由于这里是循环队列,所以相比于普通的队列,这里需要一个rear自增时需要使其能够循环回0下标处.(重点)

什么?要求设计一个循环队列?

此时rear=4,如果我们进行 %周期 操作
(rear++) % (k + 1)
= 5 % 5
=0
这样,rear就可以重新从0开始循环了.

代码实现:

bool bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {
    if (myCircularQueueIsFull(obj))
    {
        return false;
    }
    obj->date[obj->rear] = value;
    //队尾++,注意考虑回0情况.
    obj->rear = (obj->rear + 1) % (obj->k + 1);
    return true;
}

2.4 出队(myCircularQueueDeQueue)

步骤解析:

  1. 队列之前要考虑队列是否为空,队列为空返回false.
  2. front(队首下标)向后移动一位.

由于是循环队列,front也要考虑特殊情况,也需要能够回0(%周期)操作.
什么?要求设计一个循环队列?

2.5 取队首、队尾元素

队首元素很简单获取,返回obj->date[obj->front]即可.
需要注意的是:如果循环队列为空,这里规定队首返回值-1;(题干有要求).

队尾元素获取稍微复杂一些,因为存在特殊情况,如下图:
什么?要求设计一个循环队列?
此时可以直接返回obj->date[rear-1] 吗?
那岂不是date[-1]了,所以我们需要对rear进行处理.
rear - 1 + k + 1加上一套周期,那么:

0 - 1 + 5 % 5 = 4
似乎是满足要求的.

可是,不要高兴的太早了,我们为了解决这一特殊情况进行了==+周期==,那普通情况呢?
什么?要求设计一个循环队列?
rear - 1 + k + 1加上一套周期还对吗?

2 - 1 + 5 = 6.

所以我们还需要进行==%周期==操作.
即完整的:obj->date[(obj->rear - 1 + obj->k + 1) % (obj->k + 1)];

int myCircularQueueFront(MyCircularQueue* obj) {
    if (myCircularQueueIsEmpty(obj))//规定,如果队列为空,则队首是-1;
    {
        return -1;
    }
    return obj->date[obj->front];
}

int myCircularQueueRear(MyCircularQueue* obj) {
    if (myCircularQueueIsEmpty(obj))//规定,如果队列为空,则队首是-1;
    {
        return -1;
    }
    return obj->date[(obj->rear - 1 + obj->k + 1) % (obj->k + 1)];
}

代码实现:

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

2.6 循环队列的判空、判满:

在设计循环队列的时候就考虑过这个问题,所以相信大家解决这两个接口还是很简单的吧!

判空:
front(队首)和rear (待插入)指向相等时,为空.

判满:

front(队首)和rear (待插入)的下一个相等时为满.(注意%周期哦).

bool myCircularQueueIsEmpty(MyCircularQueue* obj) {
    if (obj->rear == obj->front)
    {
        return true;
    }
    else
    {
        return false;
    }
}

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

循环队列的销毁:

只需要将之前在堆区申请的两次空间释放即可.

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

什么?要求设计一个循环队列?文章来源地址https://www.toymoban.com/news/detail-467277.html

三、总代码:

#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>

typedef int Queuedate;
typedef struct {
    Queuedate* date;
    int front;//队首下标
    int rear;//待插入位置的下标
    int k;//队列的长度(固定的)
} MyCircularQueue;
bool myCircularQueueIsEmpty(MyCircularQueue* obj);
bool myCircularQueueIsFull(MyCircularQueue* obj);

MyCircularQueue* myCircularQueueCreate(int k) {
    MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));;
    obj->date = (Queuedate*)malloc(sizeof(Queuedate) * (k + 1));
    if (obj->date == NULL)
    {
        perror("obj malloc error");
    }
    obj->front = obj->rear = 0;
    obj->k = k;
    return obj;
}


bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {
    if (myCircularQueueIsFull(obj))
    {
        return false;
    }
    obj->date[obj->rear] = value;
    //队尾++
    obj->rear = (obj->rear + 1) % (obj->k + 1);
    return true;
}

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

int myCircularQueueFront(MyCircularQueue* obj) {
    if (myCircularQueueIsEmpty(obj))//如果队列为空,则队首是-1;
    {
        return -1;
    }
    return obj->date[obj->front];
}

int myCircularQueueRear(MyCircularQueue* obj) {
    if (myCircularQueueIsEmpty(obj))//如果队列为空,则队首是-1;
    {
        return -1;
    }
    return obj->date[(obj->rear - 1 + obj->k + 1) % (obj->k + 1)];
}

bool myCircularQueueIsEmpty(MyCircularQueue* obj) {
    if (obj->rear == obj->front)
    {
        return true;
    }
    else
    {
        return false;
    }
}

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

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


到了这里,关于什么?要求设计一个循环队列?的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【刷题】622. 设计循环队列

    622. 设计循环队列 设计你的循环队列实现。 循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环。它也被称为“环形缓冲器”。 循环队列的一个好处是我们可以利用这个队列之前用过的空间。在一个普通队列里,一

    2024年02月05日
    浏览(41)
  • 622. 设计循环队列(中等系列)

    设计你的循环队列实现。 循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环。它也被称为“环形缓冲器”。 循环队列的一个好处是我们可以利用这个队列之前用过的空间。在一个普通队列里,一旦一个队列满了,

    2024年02月11日
    浏览(35)
  • leetcode 622.设计循环队列

    🌟 leetcode链接:设计循环队列 1️⃣ 思路与图解: 首先循环队列是一个固定长度的队列,如果用链表的实现的话,当队列需要取队尾的数据,则需要遍历找到尾的前一个, 所以循环队列比较优的方案是用数组来实现。 又引出的问题是,如果当数据为空那么 head 指针和 tail

    2024年02月13日
    浏览(35)
  • 力扣设计循环队列

    目录 1.使用了数组来表达循环 2.循环队列是否为空 3.循环队列是否已满。 4.初始化 5.向循环队列插入一个元素。如果成功插入则返回真                          6.从队首获取元素。如果队列为空,返回 -1                                  7.获取队尾元素。如果队列为

    2024年02月07日
    浏览(34)
  • 力扣622:设计循环队列(中等)

    目录 思考 一,循环队列的初始化myCircularQueueCreate 二,判断是否为空myCircularQueueIsEmpty 三,判断队列是否满了bool myCircularQueueIsFull 四,队列的插入myCircularQueueEnQueue 五,队列的删除myCircularQueueDeQueue 六,队列取头元素和尾元素myCircularQueueFront   //   myCircularQueueRear 七,销毁队列

    2024年03月13日
    浏览(38)
  • leetcode——设计循环队列

    设计循环队列 这个题目在这里小编只分享一个解题思路,因为还有一个思路小编还在尝试,一直过不了,还在这里不断尝试,等我试出来的时候我在分享给大家,首先我们在这里给出的是数组的形式,后面在分享单链表的思路,因为数组在内存上是连续的,这里给出的思路是

    2024年02月05日
    浏览(38)
  • 栈和队列(二) 队列的实现,用栈实现队列,用队列实现栈,设计循环队列

    这里的队列我们使用链式队列,好处就是可以很方便的取出队头的元素。 使用顺序队列取出队头元素所花费的时间复杂度为O(N),把后面的元素向前移动一个下标所花费的时间。 链式队列的存储结构: 接口函数的实现 ` leetcode做题链接 主要思想:用两个队列来实现一个栈

    2024年02月11日
    浏览(39)
  • 数据结构OJ:设计循环队列

    本题为LeetCode上的经典题目,题目要求我们设计一种循环队列,满足FIFO原则且队尾被连接在队首之后。 题目中介绍循环队列的好处是可以重复利用空间,所以我们很容易想到在初始化时即开辟指定大小的空间,之后便不需要再开辟空间,只需后续销毁即可。 首先我们要选择

    2024年04月17日
    浏览(66)
  • 【数据结构与算法】设计循环队列

      🧑‍🎓 个人主页:简 料   🏆 所属专栏:C++   🏆 个人社区:越努力越幸运社区   🏆 简       介: 简料简料,简单有料~在校大学生一枚,专注C/C++/GO的干货分享,立志成为您的好帮手 ~ C/C++学习路线 (点击解锁) ❤️ C语言阶段(已结束) ❤️ 数据结构与算法(ing) ❤

    2024年01月17日
    浏览(45)
  • 【数据结构与算法】用队列实现栈&&用栈实现队列&&设计循环队列

    🌠 作者:@ 阿亮joy. 🎆 专栏:《数据结构与算法要啸着学》 🎇 座右铭:每个优秀的人都有一段沉默的时光,那段时光是付出了很多努力却得不到结果的日子,我们把它叫做扎根 请你仅使用两个队列实现一个后入先出(LIFO)的栈,并支持普通栈的全部四种操作(push、top、

    2024年01月20日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包