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

news2024/11/24 20:48:16

在这里插入图片描述

🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨
🐻推荐专栏:
🍔🍟🌯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.coloradmin.cn/o/596225.html

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

相关文章

PortSwigger 基于不安全的反序列化漏洞

一、反序列化漏洞简单介绍 反序列化漏洞是指攻击者通过在应用程序中注入恶意的序列化对象来利用应用程序的反序列化功能&#xff0c;从而导致应用程序受到攻击的漏洞。 在一些编程语言和应用程序中&#xff0c;对象可以被序列化为一些字节流或字符串&#xff0c;然后在不同的应…

基于Java+SpringBoot+Vue前后端分离网课在线学习观看系统

博主介绍&#xff1a;✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专…

C#正则表达式的使用

C#正则表达式 System.Text.RegularExpressions.Regex 使用时需要引入命名空间 using System.Text.RegularExpressions; 如果不引用则写成 System.Text.RegularExpressions.Regex 使用方法如下&#xff1a; string str"测试123456"; string result""; re…

chatgpt赋能python:Python代码中的符号

Python代码中的符号 Python是一种简单易学的编程语言&#xff0c;拥有着广泛的应用领域&#xff0c;比如数据分析、人工智能、Web开发等等。在Python的编程过程中&#xff0c;符号是我们必须要熟悉的一部分。在本文中&#xff0c;我们将介绍Python代码中常见的符号&#xff0c…

华为OD机试真题B卷 Java 实现【人民币转换】,附详细解题思路

一、题目描述 考试题目和要点&#xff1a; 中文大写金额数字前应标明“人民币”字样。中文大写金额数字应用壹、贰、叁、肆、伍、陆、柒、捌、玖、拾、佰、仟、万、亿、元、角、分、零、整等字样填写。中文大写金额数字到“元”为止的&#xff0c;在“元”之后&#xff0c;应…

【Python PyInstaller】零基础也能轻松掌握的学习路线与参考资料

一、Python PyInstaller介绍 Python PyInstaller是一个用于将Python应用程序打包成可执行文件的工具&#xff0c;支持Windows、Mac OS X和Linux平台。使用PyInstaller可以方便地将Python应用程序和所需的依赖项&#xff08;包括Python解释器本身&#xff09;打包成一个独立的可…

JS的DOM对象获取元素

测试1 getElementById <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdev…

I.MX RT1170:如何在SRAM/SDRAM运行程序

一般Flash为non-XIP时&#xff0c;我们需要在RAM上运行程序。还有一种情况&#xff0c;就是我们不想每次调试都要将程序写入Flash&#xff0c;然后由BootROM进行代码的拷贝和跳转&#xff0c;这样可以减少Flash的烧写次数。本篇文章就来讨论一下如何实现这两种情形的RAM代码运行…

chatgpt赋能python:如何更好地理解Python代码

如何更好地理解Python代码 引言 Python是一种高级编程语言&#xff0c;它越来越受欢迎。由于Python内置的强大功能和易学性&#xff0c;许多开发者选择使用Python来开发应用程序。但是&#xff0c;有时候我们可能会面临一些难以理解的代码&#xff0c;尤其是在阅读其他人的代…

一天吃透Spring面试八股文

内容摘自我的学习网站&#xff1a;topjavaer.cn Spring是一个轻量级的开源开发框架&#xff0c;主要用于管理 Java 应用程序中的组件和对象&#xff0c;并提供各种服务&#xff0c;如事务管理、安全控制、面向切面编程和远程访问等。它是一个综合性框架&#xff0c;可应用于所有…

解决Wsl2中Ubuntu无法更新软件的问题

本文排版不太好&#xff0c;详情可见笔记 有道云笔记 安装wsl2之后&#xff0c;在Ubuntu中更新软件&#xff0c;执行apt-get update命令报错&#xff0c;如下 rootjiangcheng01:~# sudo apt-get update Ign:1 http://mirrors.aliyun.com/ubuntu groovy InRelease Ign:2 http:…

一个注解的事儿,数据脱敏解决了

目录 什么是数据脱敏开整使用 Hutool 工具类实现数据掩码Hutool 信息脱敏工具类使用 Jackson 进行数据序列化脱敏 注解实现数据脱敏1、定义一个注解2、创建一个枚举类3、创建我们的自定义序列化类4、测试 项目 pom 文件 总结 本文主要分享什么是数据脱敏&#xff0c;如何优雅的…

chatgpt赋能python:Python从接口获取数据的重要性

Python从接口获取数据的重要性 在当今信息技术高速发展的时代&#xff0c;获取和分析数据是商业成功的关键因素之一。而在获取数据方面&#xff0c;使用Python编写接口来获取数据、自动化处理和分析数据的方法不仅更加高效&#xff0c;而且还可以把重点放在分析数据本身上&…

Element UI el-table form 表单的封装思路

Element UI el-table form 表单的封装思路 思路一: 将所有 form 表单类型都在 el-column-item 组件中定义好,然后根据传入的属性控制展示哪一个 form 表单类型 部分代码如下: <template><div><el-row style"padding: 0 20px"><el-col :span&…

2023年湖北七大员是哪七大员?七大员考试报名网站是哪里?启程别

2023年湖北七大员是哪七大员&#xff1f;七大员考试报名网站是哪里&#xff1f;启程别 住建厅七大员分类&#xff1a; 施工员&#xff08;土建&#xff0c;装饰装修&#xff0c;设备安装&#xff0c;市政工程&#xff09; 质量员&#xff08;土建&#xff0c;装饰装修&#x…

「从零入门推荐系统」20:推荐系统的未来发展

作者 | gongyouliu 编辑 | gongyouliu 随着科学技术的进步&#xff0c;信息技术、网络技术及物联网的快速发展&#xff0c;新信息的生产与传播更加便捷、快速。特别是最近大火的chatGPT、大模型技术引领的新一轮科技革命&#xff0c;让每一个人都可以轻松地生产各种各样的内容&…

ThreeJS教程:精灵模型Sprite作为标签

推荐&#xff1a;将 NSDT场景编辑器 加入你的3D工具链 3D工具集&#xff1a; NSDT简石数字孪生 精灵模型Sprite作为标签 实际开发的时候&#xff0c;可以使用精灵模型Sprite 颜色贴图作为标签&#xff0c;标注三维场景。 下面具体知识点&#xff0c;在精灵模型章节基本都讲解…

【多线程】阻塞队列

1. 认识阻塞队列和消息队列 阻塞队列也是一个队列&#xff0c;也是一个特殊的队列&#xff0c;也遵守先进先出的原则&#xff0c;但是带有特殊的功能。 如果阻塞队列为空&#xff0c;执行出队列操作&#xff0c;就会阻塞等待&#xff0c;阻塞到另一个线程往阻塞队列中添加元素(…

I.MX RT1170启动详解:Boot配置、Bootable image头的组成

文章目录 1 基础知识2 BOOT配置2.1 BOOT_CFG配置2.2 BOOT_MODE 3 Bootable image3.1 文件格式3.2 Bootable image头的组成3.3 Bootable image的生成3.4 例&#xff1a;BootROM之non-XIP加载过程3.5 例&#xff1a;bin文件分析 1 基础知识 &#xff08;1&#xff09;BootROM Bo…

遥感云大数据在灾害、水体与湿地领域典型案例及GPT模型教程

详情点击链接&#xff1a;遥感云大数据在灾害、水体与湿地领域典型案例及GPT模型教程 一&#xff1a;平台及基础开发平台 GEE平台及典型应用案例&#xff1b; GEE开发环境及常用数据资源&#xff1b; ChatGPT、文心一言等GPT模型 JavaScript基础&#xff1b; GEE遥感云重…