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

news/2024/10/23 9:21:14/

在这里插入图片描述

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

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

目录

  • 一、题目介绍:
    • 需要实现的接口介绍:
  • 二、接口函数的分析:
    • 2.1 循环队列的结构
    • 2.2 初始化"循环队列"(myCircularQueueCreate)
    • 2.3 入队(myCircularQueueEnQueue)
    • 2.4 出队(myCircularQueueDeQueue)
    • 2.5 取队首、队尾元素
    • 2.6 循环队列的判空、判满:
    • 循环队列的销毁:
  • 三、总代码:

一、题目介绍:

先声明一下:
题目来源:力扣(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);
}

在这里插入图片描述

三、总代码:

#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);
}

http://www.ppmy.cn/news/205773.html

相关文章

黑鲨手机全面屏的导航栏适配

我在做黑鲨k3底部导航栏适配的时候&#xff0c;用了网上的一些通用的获取底部导航栏的方法是不行的&#xff0c;试了很多&#xff0c;最终找到了合适的方法&#xff0c;来判断是否存在底部导航 因为黑鲨的系统是基于miui,所以一些底层的设置用的是和小米的类似 private static…

酷鲨商城登录页面

代码 <!DOCTYPE html> <html> <head><meta charset"UTF-8"><!-- import CSS --><!--样式文件--><link rel"stylesheet" href"https://cdn.staticfile.org/element-ui/2.15.9/theme-chalk/index.css">…

黑鲨会升级鸿蒙吗,黑鲨游戏手机2系统更新好吗?

黑鲨手机2的外观设计&#xff0c;用一句话形容就是“有延续也有回归”&#xff0c;在其身上能看到一些黑鲨Helo的元素&#xff0c;比如背部的LOGO灯、两侧的灯带&#xff0c;以及逐渐平整的后盖。而回归则是造型轮廓和手感&#xff0c;告别了黑鲨Helo的棱角分明&#xff0c;回到…

黑鲨装机大师计算机产品密钥,黑鲨装机大师U盘重装win10系统详细教程

原标题&#xff1a;黑鲨装机大师U盘重装win10系统详细教程 黑鲨装机大师是一款功能强大的装机工具。集一键重装系统、系统备份还原&#xff0c;U盘PE制作安装功能于一身&#xff0c;傻瓜式操作&#xff0c;不懂电脑也能装系统。电脑运行慢了&#xff0c;卡了&#xff0c;不求人…

黑鲨手机出现要启动android,黑鲨手机死机解决办法详细说明

不管用什么手机&#xff0c;都有可能会出现死机的情况&#xff0c;尤其是深受游戏喜好着青睐的黑鲨手机死机的时候我们又该怎么办呢&#xff1f; 1、手机电池电量过低导致的手机开不了机 因手机电池电量过低导致的手机开不了机是最常见的&#xff0c;不少网友将手机电量使用到严…

黑鲨1 救黑砖 9008救砖

目前只使用这个型号的 以后会陆续更新 黑鲨一代 适用只能进fastboot或者是黑砖的用户 进入9008&#xff1a;拆机短接或者手机关机不插线连接电脑&#xff0c;然后同时按住手机的音量上和下键不松开&#xff0c;不要按开机键&#xff0c;再来插线连接电脑。 Fastboot模式进入…

你的手机今天又被骗走了多少钱?

你的手机今天又被骗走了多少钱&#xff1f; 【摘要】 今天试着给移动发短信&#xff0c;竟然有三个定制服务。不知不觉每月都被扣去6元多&#xff01;“大意了、大意了”——只好安慰自己。同时把这篇难得一见的好文转出去&#xff0c;转给所有我认识的人&#xff0c;并附送…

黑苹果适合什么用途?_黑鲨六大配件:用途各不同,苹果安卓都能用

7月31日&#xff0c;小米生态链企业黑鲨正式推出了腾讯黑鲨游戏手机3S&#xff0c;在这场发布会上除了游戏手机以外&#xff0c;其也同时发布了一些关于手机周边的游戏外设&#xff0c;并且苹果安卓都可以用&#xff0c;兼容性不错&#xff0c;所以这篇文章我不聊手机&#xff…