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

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

题目介绍

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

思路讲解

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

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

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

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

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

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

参考代码

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;
    else
        return false;
}

bool myCircularQueueIsFull(MyCircularQueue* obj) {
    assert(obj);

    if((obj->rear + 1) % (obj->k + 1) == obj->front)
        return true;
    else
        return 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);
}

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

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

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

相关文章

  • 【数据结构】如何设计循环队列?图文解析(LeetCode)

    LeetCode链接:622. 设计循环队列 - 力扣(LeetCode) 目录 做题思路 只开辟 k 个空间 多开一个空间 代码实现 1. 循环队列的结构 2. 开辟空间 3. 判断空 4. 判断满 5. 队尾插入数据 6. 队头删除数据 7. 获取队头元素 8. 获取队尾元素 9. 销毁队列 全部代码 设计循环队列,使用数组或链表

    2024年02月10日
    浏览(44)
  • 数据结构——栈和队列OJ题

    欢迎来到专项提升小课堂! 今天的题目稍稍有难度哦! 但是只要用心,是难不倒同学们的! 题目链接:OJ链接 提示: 1 = x = 9; 最多调用100 次 push、pop、top 和 empty ; 每次调用 pop 和 top 都保证栈不为空; 核心思想: 用队列模拟出栈的先入后出这一特性! 解题思路: 此题可

    2024年02月11日
    浏览(40)
  • 数据结构OJ题——栈和队列

    题目描述:请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作(push、pop、peek、empty) void push(int x) 将元素 x 推到队列的末尾 int pop() 从队列的开头移除并返回元素 int peek() 返回队列开头的元素 boolean empty() 如果队列为空,返回 true ;否则,返回 fal

    2024年04月11日
    浏览(39)
  • 【Java数据结构 -- 队列:队列有关面试oj算法题】

    只允许在一端进行插入数据操作,在另一端进行删除数据操作得特殊线性表,队列是 先进先出 ,入队:进行插入操作得一端称为 队尾(rear) ,出队:进行删除操作的一端称为 队头(front) 。队列Queue是个接口, 底层通过链表实现的 。 boolean offer(E e) – 入队列 E poll() – 出队

    2024年01月25日
    浏览(47)
  • 【数据结构】栈与队列经典oj题

    🚀write in front🚀 📜所属专栏:初阶数据结构 🛰️博客主页:睿睿的博客主页 🛰️代码仓库:🎉VS2022_C语言仓库 🎡您的点赞、关注、收藏、评论,是对我最大的激励和支持!!! 关注我,关注我,关注我 , 你们将会看到更多的优质内容!!   栈两种线性表示都能实现

    2024年02月03日
    浏览(42)
  • 【Java 数据结构】队列与OJ题

    篮球哥温馨提示:编程的同时不要忘记锻炼哦! 目录 1、什么是队列?  2、初识Queue 2.1 认识一下Queue 2.2 简单使用下Queue 3、模拟实现Queue 3.1 构造方法和成员属性 3.2 offer 方法 3.3 poll 方法 3.4  peek 方法 4、队列相关的OJ题 4.1 设计循环队列 (来源:LeetCode 难度:中等)   4.2 用队列

    2024年01月22日
    浏览(40)
  • 【数据结构】——栈与队列(附加oj题详解)深度理解

    1.栈的定义 栈: 栈是仅限与在表尾进行插入或者删除的 线性表 我们把 允许一端插入和删除的一端叫做栈顶,另一端叫栈底,不含任何元素的栈叫做空栈 ,栈又叫做后进先出的线性表,简称 LIFO 结构 2.栈的理解 对于定义里面的在表尾进行插入和删除, 这里的表尾就是栈顶,

    2024年03月26日
    浏览(53)
  • 【LeetCode】【数据结构】栈与队列必刷OJ题

    👀 樊梓慕: 个人主页   🎥 个人专栏: 《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》 🌝 每一个不曾起舞的日子,都是对生命的辜负 目录 前言: 【LeetCode】20.有效的括号(栈的括号匹配问题) 【LeetCode】225.用队列实现栈 【LeetCode】232.用栈实现队列 【LeetCo

    2024年02月13日
    浏览(42)
  • 【数据结构与算法分析】使用C语言实现队列的两种(带头结点与不带头结点)链式存储,并且给出一种循环队列的设计思想

      当我们编写程序时,经常需要处理各种数据结构。队列是一种常见的数据结构,它有着广泛的应用场景。队列的基本操作包括入队和出队,应用于模拟等待队列、消息队列、计算机缓存等场合。   在实际编程中,我们可以用不同的数据结构来实现队列。本文主要介绍了

    2024年02月08日
    浏览(118)
  • 数据结构—循环队列(环形队列)

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

    2024年02月11日
    浏览(39)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包