数据结构OJ:设计循环队列

server/2024/9/23 22:38:53/

题目介绍

题目介绍
本题为LeetCode上的经典题目,题目要求我们设计一种循环队列,满足FIFO原则且队尾被连接在队首之后。

思路讲解

题目中介绍循环队列的好处是可以重复利用空间,所以我们很容易想到在初始化时即开辟指定大小的空间,之后便不需要再开辟空间,只需后续销毁即可。

首先我们要选择使用顺序表还是使用链表来实现循环队列,那么我们先对比一下两种方式的优缺点。

使用顺序表的优点主要在开辟空间方便,但缺点就是顺序表的删除元素效率很低,这也是链表相比于顺序表最大的优势,但只要我们稍加思考,就会发现这道题目中顺序表的缺点可以被完美避免,我们可以定义两个整形元素,一个指向队首,一个指向队尾(后文称为头指针和尾指针,虽然我们定义的是整型,但是在顺序表中,可以配合数组的索引,来实现指针的效果),如果需要出队列,我们只需让头指针+1,就可以不必移动后续数据而打到头删的效果,这种方法的实现得益于我们开头的分析:循环队列可以重复利用空间,我们让头指针+1后,原来头指针所指向的空间我们并不需要销毁,也不需要改变其中的内容,因为后续我们添加元素会将其覆盖。

使用链表的优点在于出队列和入队列很方便(即头删和尾插),所以大多数人一看到题目首先想到的就是使用链表来实现,但我们分析一下使用链表的缺点,就会发现,使用链表实现这道题会非常复杂。使用链表的缺点就是我们很难判断队列是已满还是为空,因为当头指针和尾指针相同时,可能是队列已满,也可能是队列为空。

通过以上分析,我们选择使用顺序表来实现循环队列。那么具体细节该如何实现呢?

此题的最优解为我们在创建顺序表时,数组的大小创建为(k + 1)的大小,让头指针指向第一个元素,尾指针指向最后一个元素的下一个空间,这样当头指针和尾指针指向相同时,代表头指针追上了尾指针,即队列为空;在结构体中定义一个整形元素,来记录数据的个数,当数据个数==k时,队列已满。

参考代码

typedef struct {int* list;int front;int rear;int k;
} MyCircularQueue;MyCircularQueue* myCircularQueueCreate(int k) {MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));if(obj == NULL){perror("malloc fail");return NULL;}obj->front = 0;obj->rear = 0;obj->k = k;obj->list = (int*)malloc(sizeof(int) * (k + 1));if(obj->list == NULL){perror("malloc fail");return NULL;}return obj;
}bool myCircularQueueIsEmpty(MyCircularQueue* obj) {assert(obj);if(obj->front == obj->rear)return true;elsereturn false;
}bool myCircularQueueIsFull(MyCircularQueue* obj) {assert(obj);if((obj->rear + 1) % (obj->k + 1) == obj->front)return true;elsereturn false;
}bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {assert(obj);if(myCircularQueueIsFull(obj))return false;obj->list[obj->rear] = value;obj->rear = (obj->rear + 1) % (obj->k + 1);return true;
}bool myCircularQueueDeQueue(MyCircularQueue* obj) {assert(obj);if(myCircularQueueIsEmpty(obj))return false;obj->front = (obj->front + 1) % (obj->k + 1);return true;
}int myCircularQueueFront(MyCircularQueue* obj) {assert(obj);if(myCircularQueueIsEmpty(obj))return -1;return obj->list[obj->front];
}int myCircularQueueRear(MyCircularQueue* obj) {assert(obj);if(myCircularQueueIsEmpty(obj))return -1;return obj->list[(obj->rear - 1 + 1 + obj->k) % (obj->k + 1)];
}void myCircularQueueFree(MyCircularQueue* obj) {assert(obj);free(obj->list);free(obj);
}

http://www.ppmy.cn/server/10827.html

相关文章

vue纯前端实现表格分页及条件查询功能

由于接口返回数据过慢&#xff0c;故而采用前端对数据进行处理分页的方法实现表格分页及条件查询。 一、表格 表格采用elementUI的el-table&#xff0c;只需要对数据data进行处理赋值即可。 <el-table:data"tableData"style"width: 100%"><el-t…

SIMRAD AP48 自动舵控制器维修用于 Continuum 自动驾驶仪系统Simrad显示器仪器深圳捷达工控维修

AP48 自动驾驶仪控制器是一款用于 Continuum 自动驾驶仪系统的高级专用控制头&#xff0c;采用现代玻璃舵造型进行了增强。 AP48 专为各种条件下的响应能力和易用性而设计&#xff0c;将大型铝制旋转控制旋钮与专用的“闪避键”配对&#xff0c;以 1 度或 10 度的增量调整左舷…

智能时代 | 合合信息Embedding模型荣获C-MTEB榜单第一

目录 前言 1. MTEB与C-MTEB 2. acge模型的优势 3. Embedding模型应用 4. 大模型发展的关键技术 结语 前言 随着人工智能的不断发展&#xff0c;大语言模型吸引着社会各界的广泛关注&#xff0c;支撑模型应用落地的Embedding模型成为业内的焦点&#xff0c;大模型的发展给…

【Ubuntu20.04】在ubuntu 中执行 systemd status 查询到的 Memory 的含义及方法

在 Ubuntu 中&#xff0c;使用 systemd 管理的服务&#xff0c;其内存相关的状态信息通常指的是服务运行时占用的内存。当您查询一个 systemd 服务的资源使用情况时&#xff0c;获取到的内存数据反映的是该服务在运行过程中实际使用的内存大小。这包括服务进程及其子进程所分配…

UE4 拍摄、保存并浏览相册

效果&#xff1a; 1.新建CameraActor类 2.修改截图保存路径 3.编写BP_Camera蓝图 注意路径 Save Image函数要在执行拍照和BeginPlay事件执行一次 按钮执行拍摄事件 3.编写UMG蓝图 技巧&#xff1a;让Index加1、减1循环赋值 4.把BP_Camera挂在玩家上

力扣经典150题第三十八题:生命游戏

目录 力扣经典150题第三十八题&#xff1a;生命游戏引言题目详解解题思路代码实现示例演示复杂度分析总结 力扣经典150题第三十八题&#xff1a;生命游戏 引言 本篇博客介绍了力扣经典150题中的第三十八题&#xff1a;生命游戏。生命游戏是约翰康威在1970年发明的细胞自动机&…

Oracle进阶(2)——物化视图案例延伸以及序列、同义词

一、物化视图 物化视图&#xff08;Materialized View&#xff09;是 Oracle 数据库中的一个对象&#xff0c;它是一个预先计算和存储的查询结果集&#xff0c;类似于视图&#xff0c;但与视图不同的是&#xff0c;物化视图会将查询结果保存在物理存储中&#xff0c;而不是动态…

54、图论-实现Trie前缀树

思路&#xff1a; 主要是构建一个trie前缀树结构。如果构建呢&#xff1f;看题意&#xff0c;应该当前节点对象下有几个属性&#xff1a; 1、next节点数组 2、是否为结尾 3、当前值 代码如下&#xff1a; class Trie {class Node {boolean end;Node[] nexts;public Node(…