leetcode 622. 设计循环链表

news2025/1/23 7:05:57

这道题讲了两种方法,第一个代码是用数组实现的,第二个是用链表实现的,希望对你们有帮助

(最好在VS自己测试一遍,再放到 leetcode上哦)

下面的是主函数(作参考),静下心来慢慢测试


 622. 设计循环链表

题目

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

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

你的实现应该支持如下操作:

  • MyCircularQueue(k): 构造器,设置队列长度为 k 。
  • Front: 从队首获取元素。如果队列为空,返回 -1 。
  • Rear: 获取队尾元素。如果队列为空,返回 -1 。
  • enQueue(value): 向循环队列插入一个元素。如果成功插入则返回真。
  • deQueue(): 从循环队列中删除一个元素。如果成功删除则返回真。
  • isEmpty(): 检查循环队列是否为空。
  • isFull(): 检查循环队列是否已满。

题目链接

力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台

文字 和 画图 分析

  1. 思考什么情况下,队列为空,队列为满

定义一个 指针head,用来存放头节点的地址,和一个指针tail,用来存放尾节点的地址(这个思路和队列的实现是一样的)

按照正常思路,大多数人会以为(队列长度为k):

当 head = tail 为空,而 tail 是第 k 个节点的时候为满,却忽略一点,这是个循环链表

以下这种情况就不成立:

通过图我们知道 head = tail 无法判断是空还是满

所以我们换一种思路:

存放 k 个元素,但是开辟(k + 1)个节点,故意留下一个节点不放元素

情况就是这样的:

我们发现:

当 head = tail 为空;

当 tail 的下一个节点 = head为满;

    2.  选用数组还是链表去做

这里两者思路我都讲(代码仅供参考,能通过,但是我个人觉得有些地方没有处理好,其实可以更完善,听思路即可)

  • 用数组(head 和 tail 就是元素下标)

a. 首先明确这本质是一个循环链表

b. 实现过程可能会遇到的问题:

问题1:

这里可以看到 tail 不可能一直加加

如果是正确的思路,此时的图应该是这样:

所以我们这里要对 tail 进行处理:

这里可以通用

tail = tail % (k + 1)(head也会出现这样的情况,同样要这么处理)

问题2:

判断为满时,我们可能会犯错误

这种情况我们非常容易知道:判断

tail + 1 == head

但是这种情况就不适用了:

所以我们要写一个通式:

(tail + 1 ) % (k + 1) == head

问题3:

找到尾元素

正常情况下的尾元素很好找

尾元素的下标就是 tail - 1

如果是这样,就不好判断了

这里我们可以用 if ,else语句做区分,

也可以写一个通式:

(tail - 1 + k + 1) % (k + 1)

即:(tail + k )%(k + 1)

  • 用链表(head 和 tail 就是头节点和 尾节点的地址)

a. 这里可以写一个循环链表

可以在初始化的时候先搭建好这个循环链表,后面再存放元素

b. 只有一个地方需要注意:

就是找尾元素(实际上,应该是tail的上一个节点)

这里选择可以记录上一个节点的地址

或者 循环找到上一个节点


代码

代码1:

typedef int SLType;
typedef struct StackList
{
    SLType* a;
    int top;
    int rear;
    int k;
}SL;//创建数组
void SLInit(SL* head, int k)
{
    assert(head);
    head->a = (SLType*)malloc((k + 1) * sizeof(SLType));
    head->top = 0;
    head->rear = 0;
    head->k = k;
}//初始化数组
void SLPush(SL* head, SLType x)
{
    assert(head);
    head->a[head->rear] = x;
    head->rear++;
}//存放元素
void SLPop(SL* head)
{
    assert(head);
    head->top++;
}//删除元素


//以上都是数组的实现

typedef struct
{
    SL q;
} MyCircularQueue;


MyCircularQueue* myCircularQueueCreate(int k)
{
    MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));
    SLInit(&obj->q, k);
    return obj;
}
bool myCircularQueueIsEmpty(MyCircularQueue* obj)
{
    SL* q1 = &obj->q;
    return  q1->rear == q1->top;
}//判断是否为空

bool myCircularQueueIsFull(MyCircularQueue* obj)
{
    SL* q1 = &obj->q;
    int a = q1->rear;
    a = (q1->rear + 1) % (q1->k + 1);
    return a  == q1->top;
}//判断是否为满

bool myCircularQueueEnQueue(MyCircularQueue* obj, int value)
{
    if (myCircularQueueIsFull(obj))
    {
        return false;
    }
    else
    {
        SLPush(&obj->q, value);
        SL* q1 = &obj->q;
         q1->rear = q1->rear % (q1->k + 1);
        return true;
    }

}//存放元素

bool myCircularQueueDeQueue(MyCircularQueue* obj)
{
    if (myCircularQueueIsEmpty(obj))
    {
        return false;
    }
    else
    {
        SLPop(&obj->q);
        SL* q1 = &obj->q;
         q1->top = q1->top % (q1->k + 1);
        return true;
    }

}//删除元素

int myCircularQueueFront(MyCircularQueue* obj)
{
    if (myCircularQueueIsEmpty(obj))
    {
        return -1;
    }
    else
    {
        return (&obj->q)->a[(&obj->q)->top];
    }
}//返回头元素

int myCircularQueueRear(MyCircularQueue* obj)
{
    if (myCircularQueueIsEmpty(obj))
    {
        return -1;
    }
    else
    {
        if ((&obj->q)->rear == 0)
         {
            return (&obj->q)->a[(&obj->q)->k];
         } 
        return(&obj->q)->a[(&obj->q)->rear - 1];
    }
}//返回尾元素


void myCircularQueueFree(MyCircularQueue* obj)
{
    free(obj);
}//销毁空间

 代码2:

typedef int QLType;
typedef struct QueueNode
{
    QLType val;
    struct QueueNode* next;
}QN;//创建节点
typedef struct StackList
{
    QN* head;
    QN* tail;

}QL;//创建队列
void  QNInit(QL* pphead, int k)
{
    pphead->head = pphead->tail = NULL;
    QN* prev = NULL;
    k = k + 1;
    while (k--)
    {
        QN* newnode = (QN*)malloc(sizeof(QN));
        if (pphead->head == NULL)
        {
           prev =  pphead->head = pphead->tail = newnode;
        }
        else
        {
            pphead->tail = newnode;
            pphead->head->next = pphead->tail;
            pphead->tail->next = prev;
            pphead->head = pphead->tail;
        }
    }
    pphead->head =pphead->tail = prev;
}//初始化并链接节点

QLType QLTop(QL* pphead)
{
    return pphead->head->val;
}//返回首元素
QLType QLtail(QL* pphead)
{
    QN* rear = pphead->head;
    while (rear->next != pphead->tail)
    {
        rear = rear->next;
    }
    return rear->val;
}//返回尾元素
void QLpush(QL* pphead, int val)
{
    pphead->tail->val = val;
    pphead->tail = pphead->tail->next;
}//存放元素
void QLPop(QL* pphead)
{
    pphead->head = pphead->head->next;
}//删除元素


//以上是链表的创建


typedef struct
{
    QL q;
} MyCircularQueue;

MyCircularQueue * myCircularQueueCreate(int k)
{
    MyCircularQueue* obj = (MyCircularQueue*)malloc(sizeof(MyCircularQueue));
    QNInit(&obj->q, k);
    return obj;
}//初始化
bool myCircularQueueIsEmpty(MyCircularQueue* obj)
{
    QL* q1 = &obj->q;
    return  q1->head == q1->tail;
}//判断是否为空

bool myCircularQueueIsFull(MyCircularQueue* obj)
{
    QL* q1 = &obj->q;
    return q1->tail->next == q1->head;
}//判断是否为满
bool myCircularQueueEnQueue(MyCircularQueue* obj, int value)
{
    if (myCircularQueueIsFull(obj))
    {
        return false;
    }
    else
    {
        QLpush(&obj->q, value);
        return true;
    }

}//存放元素

bool myCircularQueueDeQueue(MyCircularQueue* obj)
{
    if (myCircularQueueIsEmpty(obj))
    {
        return false;
    }
    else
    {
        QLPop(&obj->q);
        return true;
    }
}//删除元素

int myCircularQueueFront(MyCircularQueue* obj)
{
    if (myCircularQueueIsEmpty(obj))
    {
        return -1;
    }
    else
    {
        return QLTop(&obj->q);
    }
}//返回首元素

int myCircularQueueRear(MyCircularQueue* obj)
{
    if (myCircularQueueIsEmpty(obj))
    {
        return -1;
    }
    else
    {
        return  QLtail(&obj->q);
    }
}//返回尾元素


void myCircularQueueFree(MyCircularQueue* obj)
{
    free(obj);
}//释放空间

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1293085.html

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

相关文章

第21章网络通信

Internet 提供了大量有用的信息,很少有人能在接触过Internet后拒绝它的诱惑。计算机网络实现了多台计算机间的互联,使得它们彼此之间能够进行数据交流。网络应用程序就是在已连接的不同计算机上运行的程序,这些程序借助于网络协议&#xff0c…

孩子都能学会的FPGA:第二十四课——用FPGA和格雷码实现异步FIFO

(原创声明:该文是作者的原创,面向对象是FPGA入门者,后续会有进阶的高级教程。宗旨是让每个想做FPGA的人轻松入门,作者不光让大家知其然,还要让大家知其所以然!每个工程作者都搭建了全自动化的仿…

加载离线镜像包:在线镜像离线为tar包、tar离线镜像包加载并根据imageId打tag

第一步:在线环境压缩离线镜像: 需要两个文件,第一个是脚本文件image_offline_load.sh脚本,第二个是image_list.txt 按行 存放需要离线的镜像名称 ./image_offline_load.sh save image_list.txt output.tar第二步:在离…

Nginx 简单入门操作

前言:之前的文章有些过就不罗嗦了。 Nginx 基础内容 是什么? Nginx 是一个轻量级的 HTTP 服务器,采用事件驱动、异步非阻塞处理方式的服务器,它具有极好的 IO 性能,常用于 HTTP服务器(包含动静分离)、正向代理、反向代理、负载均衡 等等. Nginx 和 Node.js 在很多方…

Docker快速理解及简介

docker快速理解及简介 1.Docker为什么出现? 迁移一个项目时,运行文档、配置环境、运行环境、运行依赖包、操作系统发行版、内核等都需要重新安装配置,比较麻烦。 2.Docker是什么? Docker是基于Go语言实现的云开源项目。解决了运行…

Altium Designer实用系列(五)----整理并导出PCB的BOM表

一、引言 最近老师安排了一个小的任务,就是把我们项目的两个电路板BOM整合一下,要注明元器件的耐温、耐压、购买渠道等等内容。    一开始我觉得这工作内容太简单了,两分钟的事。但是当我实际开始干的时候,才发现,好…

【react】动态页面转换成html文件下载,解决样式问题

需求 今天遇到一个需求,挺恶心人的,将一个在线文档页面,可以导出成为html页面查看。 看到网上有使用fs模块,通过react的ReactDOMServer.renderToStaticMarkup将组件转成html字符串,输出文件了。 但是我尝试了&#x…

CodeSys学习笔记

文章目录 1.运动控制的两种方式1.1.SM3_CNC1.2.SM3_Robotics 2.两种运动控制方式的速度、加速度等参数的控制2.1.SM3_CNC2.2.SM3_Robotics 3.CNC的M指令的使用(实现)逻辑。4.SM3_Robotics中的坐标系5.SM3_Robotics如何实现插补并连续执行? 记…

【Linux】diff命令使用

diff命令 是一个用于比较两个文件或目录之间差异的命令。它可以显示两个文件之间的行级别差异,并以易于阅读的格式输出结果。 著者 由保罗艾格特、迈克海特尔、大卫海耶斯、理查德史泰尔曼和Len Tower撰写。 diff命令 -Linux手册页 语法 diff [选项] [文件1]…

设备间的指令通信

指令通信的概念 要进行设备和设备之间的交流就需要通过串口发送数据进行交流 而串口发送简单的数据只需要传输介质 但是要发送复杂的数据就需要介质和传输的规则了 三种应用场景 比如在上位机和mcu之间 通过上位机管理控制器 从而控制电池 单片机和单片机之间 用户输入数据到…

MySQl int(1)、int(20) 的区别到底在哪里

MySQl int(1)、int(20) 的区别到底在哪里 常思一二,便得自然… int(1)数据类型介绍 在MySQL中,INT(1) 是一种定义整数类型的数据字段,其中的数字表示显示宽度而不是存储范围。具体说,INT(1) 中的数字 1 表示显示宽度&#xff0…

VividTalk创新AI语音匹配图片技术:照片+语音=逼真说话视频!

VividTalk是一个由南京大学、阿里巴巴、字节跳动和南开大学共同开发的项目工具。它通过先进的音频到3D网格映射技术和网格到视频的转换技术,实现了高质量、逼真的音频驱动的说话头像视频生成。这一创新技术使得只需提供一张人物的静态照片和一段语音录音&#xff0c…

KeePass开源密码管理器

KeePass开源密码管理器 KeePass 是一款免费的开源密码管理器,KeePass 将密码存储为一个数据库,而这个数据库由一个主密码或密码文件锁住,也就是说我们只需要记住一个主密码,或使用一个密码文件,就可以解开这个数据库&a…

JavaSE基础50题:8. 获取一个数二进制序列中所有的偶数和奇数位,分别输出二进制序列。

概述 获取一个数二进制序列中所有的偶数和奇数位,分别输出二进制序列。 如:从右往左数 0111(如图) 偶数位:01 奇数位:11 代码 public static void main(String[] args) {Scanner scanner new Scanner(Sys…

动态规划学习——最长回文子序列,让字符串变成回文串的最小插入次数

一,最长回文串 1.题目 给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。 子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。 示例 1: 输入&…

Dockerfile详解#如何编写自己的Dockerfile

文章目录 前言编写规则指令详解FROM:基础镜像LABEL:镜像描述信息MAINTAINER:添加作者信息COPY:从宿主机复制文件到镜像中ADD:从宿主机复制文件到镜像中WORKDIR:设置工作目录 前言 Dockerfile是编写docker镜…

#HarmonyOS:装饰器UI描述---@Link

装饰器 装饰器(Decorator)是一种语法结构,用来在定义时修改类(class)的行为。 在语法上,装饰器有如下几个特征。 第一个是字符(或者说前缀)是,后面是一个表达式后面的…

机器学习算法(7)-朴素贝叶斯算法和K最近邻算法

一、说明 在在这篇文章中,我将解释两种机器学习算法,称为贝叶斯定理和 K 最近邻算法。贝叶斯定理以 18 世纪英国数学家托马斯贝叶斯的名字命名,是确定条件概率的数学公式。k 最近邻算法,也称为 KNN 或 k-NN,是一种非参…

Python基础知识-变量、数据类型(整型、浮点型、字符类型、布尔类型)详解

1、基本的输出和计算表达式: prinit(12-3) printf(12*3) printf(12/3) prinit(12-3) printf(12*3) printf(12/3) 形如12-3称为表达式 这个表达式的运算结果称为 表达式的返回值 1 2 3 这样的数字,叫做 字面值常量 - * /称为 运算符或者操作符 在C和j…

k8s中的Pod网络;Service网络;网络插件Calico

Pod网络;Service网络;网络插件Calico Pod网络 在K8S集群里,多个节点上的Pod相互通信,要通过网络插件来完成,比如Calico网络插件。 使用kubeadm初始化K8S集群时,有指定一个参数–pod-network-cidr10.18.0…