循环队列(C语言)
从今天开始我会开启一个专栏leetcode每日一题,大家互相交流代码经验,也当作我每天练习的自我回顾。第一天的内容是leetcode622.设计循环队列。
一、题目详细
设计你的循环队列实现。 循环队列是一种线性数据结构,其操作表现基于 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
提示:
- 所有的值都在 0 至 1000 的范围内;
- 操作数将在 1 至 1000 的范围内;
- 请不要使用内置的队列库。
虽然这是一道中等难度的题,但是理解了这道题的关键完全没有那么难。
二、解题要点
- 判断队满和队空的条件
- 空间开辟的大小确定
其实这俩点总归是一点,就是不要越界,因为我们这里主要是以数组的形式去实现这里的循环队列。
队满和队空的判断条件
主要的有两种方法:
- 留一个空间空置:满:rear +1 == front;空就是rear = front
- 增加一个size变量记录数据个数。空:size==0 满:size == MaxSize
如果我用第一个方法的话,就会浪费一个空间,用第二个则不会有这样的情况,我提供一些图画来帮大家理解。
队空时:

第一种方法队满时:
第二种方法队满时:
这里我给大家展示第一种代码的实现,大家可以自己去实现一下第二种方法
bool myCircularQueueIsEmpty(MyCircularQueue* queue) {return queue->front == queue->rear;
}bool myCircularQueueIsFull(MyCircularQueue* queue) {return((queue->rear + 1) % queue->capacity == queue->front);
}
我这里用queue->capacity代替了MaxSize;这里面涉及到里一个取模,为什么要这么做呢,其实是为了解决我们的第二个关键点,防止越界。
越界处理
队列的定义是FIFO(先进先出),是只允许在一段删除,在另一端插入的线性表。
- 允许插入的一端叫做队尾(rear)
- 允许删除的一端叫做队头(front)

上面是入队的动态操作,哈哈哈哈手画可能动图有点粗糙,但是我们可以看到6是无法入队的,因为我们按照前面的队满的判断条件(queue->rear+1)%queue->capacity==front,此时已经判断为满,也就是我们前面所说的会有一个空间被浪费,那我们取模在什么时候用的上呢,我现在就给大家举个例子!

如图所示,如果这里我不进行取模,我的6仍旧是无法入队的,并且我的rear还会越界。同样在出队时取模也有同样的妙处。
好啦把这俩个关键点弄懂,基本上这道题就没有问题了,接下来我给大家展示这道题的完整代码,并配上一组main函数作为自练习的测试样例,大家也可以对样例进行修改测试。
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>typedef struct {int capacity;int front;int rear;int* elements;
} MyCircularQueue;
MyCircularQueue* myCircularQueueCreate(int k) {MyCircularQueue* queue = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));if (queue == NULL){perror("malloc");return NULL;}queue->capacity = k + 1;queue->front = 0;queue->rear = 0;;queue->elements = (int*)malloc(sizeof(int) * queue->capacity);if (queue->elements == NULL){perror("malloc");free(queue);return NULL;}return queue;
}bool myCircularQueueEnQueue(MyCircularQueue* queue, int value) {if ((queue->rear + 1) % queue->capacity == queue->front){return false;}else{queue->rear = (queue->rear + 1) % queue->capacity;queue->elements[queue->rear] = value;return true;}
}bool myCircularQueueDeQueue(MyCircularQueue* queue) {if (queue->front == queue->rear){return false;}else{queue->front = (queue->front + 1) % queue->capacity;return true;}
}int myCircularQueueFront(MyCircularQueue* queue) {if (queue->rear == queue->front){return -1;}else{return queue->elements[(queue->front + 1) % queue->capacity];}
}int myCircularQueueRear(MyCircularQueue* queue) {if (queue->rear == queue->front){return -1;}else{return queue->elements[queue->rear];}
}bool myCircularQueueIsEmpty(MyCircularQueue* queue) {return queue->front == queue->rear;
}bool myCircularQueueIsFull(MyCircularQueue* queue) {return((queue->rear + 1) % queue->capacity == queue->front);
}
void myCircularQueueFree(MyCircularQueue* queue) {if (queue){free(queue->elements);free(queue);}
}int main() {MyCircularQueue* queue = myCircularQueueCreate(3); // 创建容量为3的循环队列printf("入队 1: %s\n", myCircularQueueEnQueue(queue, 1) ? "成功" : "失败");printf("当前队尾元素: %d\n", myCircularQueueRear(queue)); // 预期输出 1printf("当前队头元素: %d\n", myCircularQueueFront(queue)); // 预期输出 1printf("出队操作: %s\n", myCircularQueueDeQueue(queue) ? "成功" : "失败");printf("当前队头元素: %d\n", myCircularQueueFront(queue)); // 预期输出 -1printf("出队操作: %s\n", myCircularQueueDeQueue(queue) ? "成功" : "失败");printf("当前队头元素: %d\n", myCircularQueueFront(queue)); // 预期输出 -1printf("入队 2: %s\n", myCircularQueueEnQueue(queue, 2) ? "成功" : "失败");printf("入队 3: %s\n", myCircularQueueEnQueue(queue, 3) ? "成功" : "失败");printf("入队 4: %s\n", myCircularQueueEnQueue(queue, 4) ? "成功" : "失败");printf("入队 5: %s\n", myCircularQueueEnQueue(queue, 5) ? "成功" : "失败"); // 预期失败,因为队列已满// 释放队列内存myCircularQueueFree(queue);return 0;}
运行结果如下图:

有什么问题欢迎大家留言!当看到这里啦,给个小心心吧
相关文章:
循环队列(C语言)
从今天开始我会开启一个专栏leetcode每日一题,大家互相交流代码经验,也当作我每天练习的自我回顾。第一天的内容是leetcode622.设计循环队列。 一、题目详细 设计你的循环队列实现。 循环队列是一种线性数据结构,其操作表现基于 FIFO&#…...
数据可视化:让数据讲故事的艺术
目录 1 前言2 数据可视化的基本概念2.1 可视化的核心目标2.2 传统可视化手段 3 数据可视化在知识图谱中的应用3.1 知识图谱的可视化需求3.2 知识图谱的可视化方法 4 数据可视化叙事:让数据讲故事4.1 叙事可视化的关键要素4.2 数据可视化叙事的实现方法 5 数据可视化…...
雷电9最新版安装Magisk+LSPosd(新手速通)
大家好啊!我是NiJiMingCheng 我的博客:NiJiMingCheng 在安卓系统的定制与拓展过程中,获取 ROOT 权限以及安装各类框架是进阶玩家常用的操作,这可以帮助我们实现更多系统层面的个性化功能。今天,我将为大家详细介绍如何…...
Ubuntu 24.04 LTS 开启 SMB 服务,并通过 windows 访问
Ubuntu 24.04 LTS 背景资料 Ubuntu服务器折腾集Ubuntu linux 文件权限Ubuntu 空闲硬盘挂载到 文件管理器的 other locations Ubuntu开启samba和window共享文件 Ubuntu 配置 SMB 服务 安装 Samba 确保 Samba 已安装。如果未安装,运行以下命令进行安装ÿ…...
使用Websocket进行前后端实时通信
1、引入jar,spring-websocket-starter <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId> </dependency> 2、配置websocket config import org.springframe…...
vue2使用flv.js在浏览器打开flv格式视频
组件地址:GitHub - bilibili/flv.js: HTML5 FLV Player flv.js 仅支持 H.264 和 AAC/MP3 编码的 FLV 文件。如果视频文件使用了其他编码格式就打不开。 flv.vue <template><div><el-dialog :visible.sync"innerVisibleFlv" :close-on-pre…...
OpenCV相机标定与3D重建(61)处理未校准的立体图像对函数stereoRectifyUncalibrated()的使用
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 为未校准的立体相机计算一个校正变换。 cv::stereoRectifyUncalibrated 是 OpenCV 库中的一个函数,用于处理未校准的立体图像对。该函…...
[cg] glProgramBinary
参考: glProgramBinary - OpenGL 4 Reference Pages opengl 通过gpu编译好的 shader 可以存储到二进制文件中,第二次使用的时候直接加载二进制文件即可, glProgramBinary用于加载shader的二进制数据 实列代码如下: // 假设已经…...
LeetCode hot 力扣热题100 二叉树的最大深度
class Solution { public:int maxDepth(TreeNode* root) {if (root nullptr) {return 0;}int l_depth maxDepth(root->left);int r_depth maxDepth(root->right);return max(l_depth, r_depth) 1;} }; 代码作用 该函数通过递归计算二叉树的最大深度(从根节…...
速通Docker === 网络
目录 Docker网络详解 容器之间直接通信的弊端 (一)启动容器 (二)进入容器并发起请求 (三)请求流程 (四) 弊端分析 一、Docker网络基础 (一)容器IP分配…...
【MySQL — 数据库基础】深入解析MySQL常用数据类型
常用数据类型 创建完数据库之后,就要在数据库中创建表,表中存储的数据记录,一条记录由不同的列组成,每条列都需要自己的类型;并且表中的多个行对应的列的数据类型,都必须是相同的; 那么每个…...
Linux高级--3.3.1 C++ spdlog 开源异步日志方案
一、基本介绍 spdlog 是由 Gustav S. 在 2015 年开发的一个高性能 C 日志库。开发这个库的主要目的是为了提供一个非常快速、轻量、易于使用的日志工具,特别适合需要高性能、低延迟日志记录的 C 应用程序。(由于源码现在比较难下载,我把压缩…...
电梯系统的UML文档05
Dispatcher 不控制实际的电梯组件,但它在软件系统中是重要的。每一个电梯有一个ispatcher,主要功能是计算电梯的移动方向、移动目的地以及保持门的打开时间。它和系统中除灯控制器以外的几乎所有控制对象交互。 安全装置也是一个环境对象,它…...
如何使 LLaMA-Factory 支持 google/gemma-2-2b-jpn-it 的微调
如何使 LLaMA-Factory 支持 google/gemma-2-2b-jpn-it 的微调 追加, "Gemma-2-2B-JPN-Instruct": {DownloadSource.DEFAULT: "google/gemma-2-2b-jpn-it",},修改 constants.py, vi ./src/llamafactory/extras/constants.py---"…...
MySQL中日期和时间戳的转换:字符到DATE和TIMESTAMP的相互转换
在MySQL中,经常需要在 DATE、TIMESTAMP 和字符串之间进行相互转换。以下是一些常见的转换方法: 1. 字符串到日期/时间类型 字符串转 DATE: 使用 STR_TO_DATE() 函数将字符串转换为 DATE 类型。你需要提供字符串的格式。 SELECT STR_TO_DATE(2024-08-24,…...
HarmonyOS NEXT开发进阶(十):UIAbility 组件交互
文章目录 一、前言二、启动应用内的 UIAbility三、启动应用内的UIAbility并获取返回结果四、启动其他应用的UIAbility五、启动其他应用的 UIAbility 并获取返回结果六、启动 UIAbility 的指定页面6.1 调用方 UIAbility 指定启动页面6.2 目标 UIAbility 首次启动6.3 目标UIAbili…...
深入探索Math.NET:开启高效数值计算之旅
一、引言 在当今数字化时代,数值计算已然成为科学研究、工程设计、金融分析等众多领域的核心驱动力。从探索宇宙奥秘的物理学计算,到优化建筑结构的土木工程设计,再到预测市场趋势的金融建模,数值计算的身影无处不在,…...
AI编程工具横向评测--Cloudstudio塑造完全态的jupyter notebook助力数据分析应用开发
AI编程工具横向评测–Cloudstudio塑造完全态的jupyter notebook助力数据分析应用开发 数据分析类应用的开发,指的是首先进行数据分析,比如统计学分析、机器学习模型的构建等,然后将分析的流程开发成数据分析类的工具,或者将数据分…...
【2024 CSDN博客之星】技术洞察类:从DeepSeek-V3的成功,看MoE混合专家网络对深度学习算法领域的影响(MoE代码级实战)
目录 一、引言 1.1 本篇文章侧重点 1.2 技术洞察—MoE(Mixture-of-Experts,混合专家网络) 二、MoE(Mixture-of-Experts,混合专家网络) 2.1 技术原理 2.2 技术优缺点 2.3 业务代码实践 2.3.1 业务场…...
Linux——入门基本指令汇总
目录 1. ls指令2. pwd3. whoami指令4. cd指令5. clear指令6. touch指令7. mkdir指令8. rm指令9. man指令10. cp指令11. mv指令12. cat指令13. tac指令14. more指令15. less指令16. head指令17. tail指令18. date指令19. cal指令20. find指令21. which指令22. alias指令23. grep…...
LLM API 防降智!IMMACULATE 框架,1% 开销搞定审计验证
来源:机器之心 本文约2500字,建议阅读5分钟本文介绍了 IMMACULATE 框架,可低开销审计黑盒 LLM API 违规行为。本文作者分别来自新加坡国立大学和加州大学伯克利分校。第一作者郭衍培来自新加坡国立大学,长期关注大语言模型基础设施…...
QTableWidget 表格组件耙
7.1 初识三维模型 7.1.1 三维模型的数据载体 随着计算机图形技术的发展,我们或多或少都会见过或者听说过三维模型。笔者始终记得小时候第一次在电视上看到三维动画《变形金刚:超能勇士》的震撼感受;而现在我们已经可以在手机上玩三维游戏《王…...
PowerToys MeasureTool:让屏幕测量变得如此简单,设计师必备的免费神器
PowerToys MeasureTool:让屏幕测量变得如此简单,设计师必备的免费神器 【免费下载链接】PowerToys Microsoft PowerToys is a collection of utilities that supercharge productivity and customization on Windows 项目地址: https://gitcode.com/Gi…...
OpenClaw定时任务管理:千问3.5-27B实现凌晨自动备份
OpenClaw定时任务管理:千问3.5-27B实现凌晨自动备份 1. 为什么需要AI驱动的定时任务? 上个月我经历了一次惨痛的数据丢失——连续三天熬夜写的代码,因为笔记本突然蓝屏而全部消失。虽然最终通过碎片文件恢复了部分内容,但这件事…...
Bugku普通的二维码、薛定谔的猫
普通的二维码解压文件夹是一个二维码扫描得到用010 Editor十六进制文本编辑器打开发现一串数字(仅有0-7,推测是八进制)14615414114717311014116614513717106012513712017113716314316215116016413711716414313712415713712414515613710116314…...
Generalist最新长文定调:具身原生才是正道,中国玩家原力灵机已交卷
Jay 发自 凹非寺量子位 | 公众号 QbitAIGeneralist AI的GEN-1热度,仍在发酵。自节前那场引爆全网的Demo之后,昨日,创始人Pete Florence与团队,正式释出了GEN-1的技术博客。与其说这是一篇技术分享,不如说这是一篇「教同…...
Triton + RISC-V缓
. GIF文件结构 相比于 WAV 文件的简单粗暴,GIF 的结构要精密得多,因为它天生是为了网络传输而设计的(包含了压缩机制)。 当我们用二进制视角观察 GIF 时,它是由一个个 数据块(Block) 组成的&…...
Blazor Hybrid跨端失控?揭秘WinUI3/MacCatalyst/iOS 18原生桥接的3种反模式与1套工业级Bridge Protocol设计规范
第一章:Blazor Hybrid跨端失控的本质与2026技术拐点研判Blazor Hybrid 的“跨端失控”并非架构缺陷,而是其运行时契约在多宿主环境(WebView2、Android WebView、iOS WKWebView)中持续弱化的必然结果。当 .NET MAUI 或 Avalonia 作…...
为什么92%的PHP团队异步化失败?——高并发场景下I/O等待、内存泄漏与协程调度三大暗礁全曝光
第一章:PHP异步I/O的底层本质与失败全景图PHP 传统同步阻塞模型在 I/O 密集型场景中天然受限——每次 socket read/write、数据库查询或 HTTP 请求都会让整个进程挂起,直至内核返回结果。其底层本质并非缺乏异步能力,而是运行时(Z…...
[AI/应用/MCP] MCP Server/Tool 开发指南蛊
简介 langchain专门用于构建LLM大语言模型,其中提供了大量的prompt模板,和组件,通过chain(链)的方式将流程连接起来,操作简单,开发便捷。 环境配置 安装langchain框架 pip install langchain langchain-community 其中…...
