【数据结构】_队列经典算法OJ:循环队列

目录

[1. 题目描述及链接](#1. 题目描述及链接)

[2. 解题思路](#2. 解题思路)

[2.1 循环队列的设计](#2.1 循环队列的设计)

[2.2 关于循环队列的假溢出问题](#2.2 关于循环队列的假溢出问题)

[3. 程序](#3. 程序)

注:部分方法实现细节


1. 题目描述及链接

题目链接:622. 设计循环队列 - 力扣(LeetCode)

题目描述:

设计你的循环队列实现。 循环队列是一种线性数据结构,其操作表现基于 FIFO(先进先出)原则并且队尾被连接在队首之后以形成一个循环。它也被称为"环形缓冲器"。

循环队列的一个好处是我们可以利用这个队列之前用过的空间。在一个普通队列里,一旦一个队列满了,我们就不能插入下一个元素,即使在队列前面仍有空间。但是使用循环队列,我们能使用这些空间去存储新的值。

2. 解题思路

2.1 循环队列的设计

若采用数组作为队列底层结构,则需创建head标记数组第一个元素,tail标记数组最后一个元素的下一个位置,由于下标不具有循环特性,循环队列实现的麻烦之处在于下标超过k的下标处理问题,但获取队尾元素很方便;

若采用单链表作为队列底层结构,则需创建head指向链表第一个结点,tail指向链表最后一个结点,由于链表结点的链接关系,实现增删操作无需像数组结构一样需手动处理下标循环问题,但获取尾结点遍历等操作较为麻烦;

本题采用数组作为底层结构;

2.2 关于循环队列的假溢出问题

假设当前数组空间为4个数据大小,即k=4,依次进行以下操作:

解决方法:

方法1:增加有效数据存储的记录变量size

每插入一个元素则size++,每删除一个元素则size--,size=0表示队列为空,size=k表示队列满;

方法2:额外多开一个空间

3. 程序

cpp 复制代码
typedef struct {
    int* a;
    int head;  // 指向第一个数据
    int tail;  // 指向最后一个数据的下一个数据
    int k;
} MyCircularQueue;
bool myCircularQueueIsEmpty(MyCircularQueue* obj) {
    return obj->head == obj->tail;
}

bool myCircularQueueIsFull(MyCircularQueue* obj) {
    return (obj->tail+1)%(obj->k+1) == obj->head;
}

MyCircularQueue* myCircularQueueCreate(int k) {
    MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));  
    // 额外开一个空间解决假溢出问题
    obj->a=(int*)malloc(sizeof(int)*(k+1));
    obj->head=0;
    obj->tail=0;
    obj->k=k;  
    return obj;
}

bool myCircularQueueEnQueue(MyCircularQueue* obj, int value) {
    if(myCircularQueueIsFull(obj)){
        return false;
    }
    obj->a[obj->tail]=value;
    obj->tail++;
    obj->tail %= (obj->k+1);
    return true;
}

bool myCircularQueueDeQueue(MyCircularQueue* obj) {
    if(myCircularQueueIsEmpty(obj)){
        return false;
    }
    obj->head++;
    obj->head %= (obj->k+1);
    return true;
}

int myCircularQueueFront(MyCircularQueue* obj) {
    if(myCircularQueueIsEmpty(obj)){
        return -1;
    }else{
        return obj->a[obj->head];
    }
}

int myCircularQueueRear(MyCircularQueue* obj) {
    if(myCircularQueueIsEmpty(obj)){
        return -1;
    }else{
        //return obj->tail==0?obj->a[obj->k]:obj->a[obj->tail-1];
        return obj->a[(obj->tail+obj->k)%(obj->k+1)];
    }
}


void myCircularQueueFree(MyCircularQueue* obj) {
    free(obj->a);
    free(obj);
}

注:部分方法实现细节

关于获取队尾元素的下标计算:

方式1:

cpp 复制代码
return obj->tail==0?obj->a[obj->k]:obj->a[obj->tail-1];

三目操作符特殊处理tail指向第一个数据位置:此时队列最后一个有效元素是a[k];

方式2:

cpp 复制代码
return obj->a[(obj->tail-1+obj->k+1)%(obj->k+1)];

对 tail-1 进行 +(k+1) 再模 (k+1) 的操作,使得tail=0时tail-1=-1的特殊情况得以统一处理,其中-1+1可抵消:

cpp 复制代码
return obj->a[(obj->tail+obj->k)%(obj->k+1)];
相关推荐
Swift社区5 分钟前
LeetCode 270:在二叉搜索树中寻找最接近的值(Swift 实战解析)
算法·leetcode·swift
mikey棒棒棒21 分钟前
二叉树的遍历与构造
算法·二叉树
喂我花生(๑•̀ㅂ•́)و✧1 小时前
JAVA中ArrayList的解析
java·开发语言·算法
摄殓永恒1 小时前
【入门】斜角II
c++·算法·图论
ccLianLian1 小时前
数据结构·ST表
数据结构
大锦终2 小时前
【C++】红黑树
c语言·开发语言·数据结构·c++
David Bates2 小时前
代码随想录第41天:图论2(岛屿系列)
python·算法·图论
似水এ᭄往昔2 小时前
【数据结构】——栈
c语言·数据结构
司小豆2 小时前
视觉-语言基础模型作为高效的机器人模仿学习范式
人工智能·算法·机器人
福居路冥想的草莓3 小时前
逆波兰表达式求值(中等)
数据结构