【Leedcode】栈和队列必备的面试题(第二期)

news2025/1/12 6:00:18

【Leedcode】栈和队列必备的面试题(第二期)


文章目录

  • 【Leedcode】栈和队列必备的面试题(第二期)
  • 一、题目(用两个队列实现栈)
  • 二、思路+图解
    • 1.定义两个队列
    • 2.初始化两个队列
    • 3.往两个队列中放入数据
    • 4.两个队列出数据
    • 5.显示栈顶信息
    • 6.判断栈或者队列是否为空
    • 7.销毁栈或者队列
  • 三、总代码展示
  • 总结


一、题目(用两个队列实现栈)

Leedcode链接


在这里插入图片描述


在这里插入图片描述


在这里插入图片描述
这几个接口使我们需要实现的我会一 一实现并讲解!


二、思路+图解

注意:这里会用到很多栈和队列接口实现的一些知识,这里不再深究,如果想了解可以去下面两个博客!
栈的接口模拟实现 + 队列的接口模拟实现


做本题需要的接口和结构体声明!

代码如下(示例):

typedef int QDataType;
typedef struct QueueNode
{
  struct QueueNode* next;
  QDataType data;
}QueueNode;

typedef struct Queue
{
 QueueNode* head;
 QueueNode* tail;
}Queue;

void QueueInit(Queue* pq);//初始化队列
void QueueDestroy(Queue* pq);//销毁队列
void QueuePush(Queue* pq, QDataType x);//放入数据
void QueuePop(Queue* pq);//删除数据
QDataType QueueFront(Queue* pq);//取头数据
QDataType QueueBack(Queue* pq);//取尾数据
size_t QueueSize(Queue* pq);//计算数据个数
bool QueueEmpty(Queue* pq);//判断队列是不是空

void QueueInit(Queue* pq)
{
   assert(pq);
   pq->head = NULL;
   pq->tail = NULL;
}

void QueueDestroy(Queue* pq)//销毁队列
{
 assert(pq);
 //创建一个cur指针 指向队列头 用于挨个释放空间
 QueueNode* cur = pq->head;

 //当cur不为NULL 一直循环执行  挨个释放列表成员空间
 while (cur != NULL)
 {
  QueueNode* next = cur->next;
  free(cur);
  cur = next;
 }

 //当循环结束 将 head与tail 置为NULL
 pq->head = pq->tail = NULL;
}

void QueuePush(Queue* pq, QDataType x)//放入数据
{
    assert(pq);
    QueueNode* newnode = (QueueNode*)malloc(sizeof(QueueNode));
    newnode->data = x;
    newnode->next = NULL; 
    if (pq->head == NULL)
     {
        pq->head = pq->tail = newnode;
     }
     else
     {
        pq->tail->next = newnode;
        pq->tail = newnode;
     }

}

void QueuePop(Queue* pq)//删除数据
{
 assert(pq);
 assert(!QueueEmpty(pq));
 QueueNode* next = pq->head->next;
 free(pq->head);
 if (pq->head == pq->tail)
 {
  pq->tail = NULL;
 }
 pq->head = next; 
}

QDataType QueueFront(Queue* pq)//取头数据
{
 assert(pq);
 assert(!QueueEmpty(pq));

 return pq->head->data;
}

QDataType QueueBack(Queue* pq)//取尾数据
{
    assert(pq);
    assert(!QueueEmpty(pq));
    return pq->tail->data;
}
size_t QueueSize(Queue* pq)//计算数据个数
{
   assert(pq);
   assert(!QueueEmpty(pq));
   QueueNode* cur = pq->head;
   int n = 0;
   while (cur)
  {
    cur = cur->next;
    n++;
  }
   return n;
}

bool QueueEmpty(Queue* pq)//判断队列是不是空
{
    assert(pq);
    return pq->head == NULL && pq->tail == NULL;
}

1.定义两个队列

代码如下(示例):

typedef struct{
    Queue q1;
    Queue q2;
} MyStack;

2.初始化两个队列

在这里插入图片描述


在这里插入图片描述


在这里插入图片描述

代码如下(示例):

MyStack* myStackCreate() {
    MyStack* st = (MyStack*)malloc(sizeof(MyStack));
    QueueInit(&st->q1);
    QueueInit(&st->q2);
    return st;
}

3.往两个队列中放入数据


在这里插入图片描述


在这里插入图片描述


代码如下(示例):

void myStackPush(MyStack* obj, int x) {
    if(!QueueEmpty(&obj->q1))
    {
        QueuePush(&obj->q1, x);
    }
    else
    {
        QueuePush(&obj->q2, x);
    }
}

4.两个队列出数据

在这里插入图片描述


在这里插入图片描述


在这里插入图片描述


在这里插入图片描述


代码如下(示例):

int myStackPop(MyStack* obj) {
    Queue* emptyQ = &obj->q1;
    Queue* nonemptyQ = &obj->q2;
    if(!QueueEmpty(&obj->q1))
    {
        nonemptyQ = &obj->q1;
        emptyQ = &obj->q2;
    }
    while(QueueSize(nonemptyQ)>1)
    {
        QueuePush(emptyQ, QueueFront(nonemptyQ));
        QueuePop(nonemptyQ);
    }
    int ret = QueueFront(nonemptyQ);
    QueuePop(nonemptyQ);
     return ret;
}

5.显示栈顶信息

在这里插入图片描述

代码如下(示例):

int myStackTop(MyStack* obj) {
    if(!QueueEmpty(&obj->q1))
    {
        return QueueBack(&obj->q1);
    }
    else
    {
        return QueueBack(&obj->q2);
    }
}

6.判断栈或者队列是否为空

在这里插入图片描述

代码如下(示例):

bool myStackEmpty(MyStack* obj) 
{
    return QueueEmpty(&obj->q1) && QueueEmpty(&obj->q2);
}

7.销毁栈或者队列

在这里插入图片描述


在这里插入图片描述

代码如下(示例):

void myStackFree(MyStack* obj) 
{
    QueueDestroy(&obj->q1);
    QueueDestroy(&obj->q2);
    free(obj);
}

三、总代码展示

代码如下(示例):

typedef int QDataType;
typedef struct QueueNode
{
  struct QueueNode* next;
  QDataType data;
}QueueNode;

typedef struct Queue
{
 QueueNode* head;
 QueueNode* tail;
}Queue;

void QueueInit(Queue* pq);//初始化队列
void QueueDestroy(Queue* pq);//销毁队列
void QueuePush(Queue* pq, QDataType x);//放入数据
void QueuePop(Queue* pq);//删除数据
QDataType QueueFront(Queue* pq);//取头数据
QDataType QueueBack(Queue* pq);//取尾数据
size_t QueueSize(Queue* pq);//计算数据个数
bool QueueEmpty(Queue* pq);//判断队列是不是空

void QueueInit(Queue* pq)
{
   assert(pq);
   pq->head = NULL;
   pq->tail = NULL;
}

void QueueDestroy(Queue* pq)//销毁队列
{
 assert(pq);
 //创建一个cur指针 指向队列头 用于挨个释放空间
 QueueNode* cur = pq->head;

 //当cur不为NULL 一直循环执行  挨个释放列表成员空间
 while (cur != NULL)
 {
  QueueNode* next = cur->next;
  free(cur);
  cur = next;
 }

 //当循环结束 将 head与tail 置为NULL
 pq->head = pq->tail = NULL;
}

void QueuePush(Queue* pq, QDataType x)//放入数据
{
    assert(pq);
    QueueNode* newnode = (QueueNode*)malloc(sizeof(QueueNode));
    newnode->data = x;
    newnode->next = NULL; 
    if (pq->head == NULL)
     {
        pq->head = pq->tail = newnode;
     }
     else
     {
        pq->tail->next = newnode;
        pq->tail = newnode;
     }

}

void QueuePop(Queue* pq)//删除数据
{
 assert(pq);
 assert(!QueueEmpty(pq));
 QueueNode* next = pq->head->next;
 free(pq->head);
 if (pq->head == pq->tail)
 {
  pq->tail = NULL;
 }
 pq->head = next; 
}

QDataType QueueFront(Queue* pq)//取头数据
{
 assert(pq);
 assert(!QueueEmpty(pq));

 return pq->head->data;
}

QDataType QueueBack(Queue* pq)//取尾数据
{
    assert(pq);
    assert(!QueueEmpty(pq));
    return pq->tail->data;
}
size_t QueueSize(Queue* pq)//计算数据个数
{
   assert(pq);
   assert(!QueueEmpty(pq));
   QueueNode* cur = pq->head;
   int n = 0;
   while (cur)
  {
    cur = cur->next;
    n++;
  }
   return n;
}

bool QueueEmpty(Queue* pq)//判断队列是不是空
{
    assert(pq);
    return pq->head == NULL && pq->tail == NULL;
}

//定义两个队列
typedef struct{
    Queue q1;
    Queue q2;
} MyStack;


MyStack* myStackCreate() {
    //函数内定义变量出了函数就没了,所以我们在这里选择用malloc开辟动态空间 并将这个空间的地址返回给函数
    MyStack* st = (MyStack*)malloc(sizeof(MyStack));
    //调用自己的队列初始化函数 将定义的2个队列初始化
    QueueInit(&st->q1);
    QueueInit(&st->q2);
    //函数要求 返回新开辟的地址(栈地址)
    return st;
}

void myStackPush(MyStack* obj, int x) {
    //往栈中放入数据
    if(!QueueEmpty(&obj->q1))//当q1不是空的时 将数据放入q1队列中
    {
        //调用队列函数,自定义函数传过来的是栈的指针,所以要&obj的地址指向q1
        QueuePush(&obj->q1, x);
    }
    else//当q2不是空的时 将数据放入q1队列中  由于这里是else 所以假设q1 q2都是空也会将数据放入q2队列中
    {
        QueuePush(&obj->q2, x);
    }
}

int myStackPop(MyStack* obj) {
    //emptyQ表示空队列   nonEmptuQ 表示有元素队列
    //先假设定义队列 q1为空 q2 非空
    Queue* emptyQ = &obj->q1;
    Queue* nonemptyQ = &obj->q2;
    //然后调用判断是否为空函数
    if(!QueueEmpty(&obj->q1))//如果队列q1 非空进入循环
    {
        //向q1道歉 将q1改为非空  将q2改为空
        nonemptyQ = &obj->q1;
        emptyQ = &obj->q2;
    }
    //此时 nonempyuQ 与 emptyQ 经历了修正 队列nonempyuQ 是非空 emptyQ是空 (特殊情况下两者都是空)
    while(QueueSize(nonemptyQ)>1)//循环条件 非空的nonemptyQ队列数据个数最少是大于1就可以进入循环,假设等于或小于1不进入循环
    {
        //将非空nonemptyQ队列元素数据放入,空emptyQ队列中
        QueuePush(emptyQ, QueueFront(nonemptyQ));
        //放进去一个就删除一个非空nonemptyQ队列元素
        QueuePop(nonemptyQ);
    }
    //当循环结束来到这里nonempty内只有一个元素,这个元素也表示的是栈顶的元素(特殊情况下nonempty也是空的)
    int ret = QueueFront(nonemptyQ);
    QueuePop(nonemptyQ);//最后删除这个元素(表达出栈行为)
     return ret;//函数要求返回这个被删除的元素数据
}
int myStackTop(MyStack* obj) {
    //显示栈顶信息
    if(!QueueEmpty(&obj->q1))//栈顶不为空进入
    {
        //返回q1队列尾部数据
        return QueueBack(&obj->q1);
    }
    else
    {
        //返回q2队列尾部数据
        return QueueBack(&obj->q2);
    }
}

bool myStackEmpty(MyStack* obj) {
    //当q1为空 并且 q2为空 则返回真 
    return QueueEmpty(&obj->q1) && QueueEmpty(&obj->q2);
}

void myStackFree(MyStack* obj) {
    //先将q1 和 q2 空间销毁
    QueueDestroy(&obj->q1);
    QueueDestroy(&obj->q2);
    //最后释放obj空间
    free(obj);
}

/**
 * Your MyStack struct will be instantiated and called as such:
 * MyStack* obj = myStackCreate();
 * myStackPush(obj, x);
 
 * int param_2 = myStackPop(obj);
 
 * int param_3 = myStackTop(obj);
 
 * bool param_4 = myStackEmpty(obj);
 
 * myStackFree(obj);
*/


总结

以上就是今天要讲的内容,本文为【Leedcode】栈和队列必备的面试题(第二期)
如果我的博客对你有所帮助记得三连支持一下,感谢大家的支持!
在这里插入图片描述

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

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

相关文章

Linux 自带按键驱动

目录 一、内核检查 二、驱动文件 三、设备树 四、验证 一、内核检查 内核一般默认已经使能了 KEY 驱动,但是还是要检查一下。按照如下路径找到相应的配置选项: Device Drivers -> Input device support -> Generic in…

WebRTC拥塞控制原理解析

WebRTC包含三种拥塞控制算法,GCC、BBR和PCC。其中,BBR一开始是针对TCP的拥塞控制提出来的。它的输入为ACK/SACK,输出为拥塞窗口(congestion_window)发送速度(pacing_rate)。BBR是怎样运用到UDP,甚至运用到实时流媒体传输之上的&am…

Java-封装、继承、多态

封装 访问控制权限又成为“封装”,是面向对象三大特征中的一种。核心是,只对需要的类可见。 继承 继承是所有OOP(Object Oriented Programming)语言和Java语言都不可或缺的一部分。 只要创建一个类,就隐式继承自Obje…

【计算机组成原理】计算机硬件的基础组成、认识各个硬件部件

计算机组成原理(一) 计算机内部是通过电信号传递数据 电信号:分为高电平和低电平,分别代表1/0 数字、文字、图像如何用二进制表示? CPU如何对二进制数进行加减乘除? 如何存储这些二进制数的? 如何从内存中取出想要的数…

RuoYi-Flowable-Plus搭建(若依)

项目简介 1.本项目基于 RuoYi-Vue-Plus 进行二次开发,采用 Flowable 扩展工作流应用场景,支持在线表单设计和丰富的工作流程设计能力。2.本项目主要针对Flowable工作流场景开发,采用MIT开源协议,完全免费给个人及企业使用。3.环境…

(小甲鱼python)函数笔记合集七 函数(IX)总结 函数的递归、递归和迭代的区别详解

一、基础复习 函数的基本用法 创建和调用函数 函数的形参与实参等等函数的几种参数 位置参数、关键字参数、默认参数等函数的收集参数*args **args 解包参数详解函数中参数的作用域 局部作用域 全局作用域 global语句 嵌套函数 nonlocal语句等详解函数的闭包(工厂函…

华为OD机试题【有效子字符串】用 C++ 进行编码 (2023.Q1)

最近更新的博客 华为od 2023 | 什么是华为od,od 薪资待遇,od机试题清单华为OD机试真题大全,用 Python 解华为机试题 | 机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为od机试,独家整理 已参加机试人员的实战技巧文章目录 最近更新的博客使用说明有效子…

如何使用New Bing的ChatGPT聊天机器人

New Bing是微软推出的一款基于GPT4模型的智能搜索引擎,可以提供更加人性化、交互式和多样化的搜索体验。在中国使用New Bing的ChatGPT聊天机器人需要以下几个步骤: 一、准备工作 1.1 申请微软邮箱 用来登录Edge浏览器和接收来自微软New Bing使用邀请的…

代码随想录算法训练营 || 贪心算法 435 763 56

Day31435. 无重叠区间力扣题目链接给定一个区间的集合,找到需要移除区间的最小数量,使剩余区间互不重叠。注意: 可以认为区间的终点总是大于它的起点。 区间 [1,2] 和 [2,3] 的边界相互“接触”,但没有相互重叠。示例 1:输入: [ [1,2], [2,3]…

TryHackMe-hackerNote

hackerNote 自定义 Web 应用程序,引入用户名枚举、自定义单词列表和基本权限提升漏洞。 端口扫描 循例 nmap Web 80和8080都是一样的页面,并且存在一个登录页面 除此之外还有另一个信息就是它只有一个用户 存在注册功能,我们注册一个用户…

AntDB数据库首个社区版正式发布,携手生态更多可能

3月1日,亚信科技AntDB数据库社区版正式开放免费下载。作为AntDB数据库的首个社区版产品,AntDB-CE(Community Edition ) V7.2.0具有完整、易用、兼容度高的企业级分布式数据库产品特性,提供了详细的在线文档支持&#x…

Kafka基本原理

总述 简介 Kafka是最初由Linkedin公司开发,是一个分布式、支持分区的(partition)、多副本的(replica),基于zookeeper协调的分布式消息系统,它的最大的特性就是可以实时的处理大量数据以满足各…

高清无线手机投屏电脑软件Airserver7全平台兼容

Airserver是一款优质的手机投屏软件,搭配了可调节的分辨率模式,针对不同的网络配置,还专门研发了低数据模式以及硬件加速功能,全方位保证投屏画面的清晰度。除此之外,通过调整画面的色彩变化,还可以增强用户…

OpenWrt介绍及编译基础教程

关于OpenWrt OpenWRT是一个高度模块化、高度自动化的嵌入式Linux系统,拥有强大的网络组件和扩展性,常常被用于工控设备、电话、小型机器人、智能家居、路由器以及VOIP设备中。 同时,它还提供了100多个已编译好的软件,而且数量还在…

SC12B触摸感应芯片评测方案(1)

MM32F0160SC12B Touch Application Evaluation 文章目录MM32F0160SC12B Touch Application EvaluationIntroduction & RequirementHardwareSC12B & SC12B Sample Demo boardMini-F0160 boardSoftwareMCU Software - MM32F0160PC Tool - FreeMASTERSummaryIntroduction …

Jupyter Lab | “丢下R,一起来快乐地糟蹋服务器!”

写作前面 工具永远只是为了帮助自己提升工作效率 —— 沃兹基硕得 所以说,为什么要使用jupyterlab呢?当然是因为基于服务器来处理数据就可以使劲造了,而且深切地感觉到,“R这玩意儿是人用的吗”。 jupyter-lab | mamba安装以及…

【微信小程序】-- 全局配置 -- window - 导航栏(十五)

💌 所属专栏:【微信小程序开发教程】 😀 作  者:我是夜阑的狗🐶 🚀 个人简介:一个正在努力学技术的CV工程师,专注基础和实战分享 ,欢迎咨询! &…

Thymeleaf -- 视图模块

添加thymeleaf的jar包2. 新建一个Servlet类ViewBaseServletpublicclassViewBaseServletextendsHttpServlet { ​privateTemplateEnginetemplateEngine; ​Overridepublicvoidinit() throwsServletException { ​// 1.获取ServletContext对象ServletContextservletContextthis.g…

人大金仓和达梦的空间数据能力对比

一、总得来说: 人大金仓底层更解决于pg数据库, 人大金仓的空间能力基于postgis能力来实现,能力挺强大的. 细节上人大金仓的架构上也对空间的支持框架做的比达梦更加完善。例如数据库的集群能力,并行计算能力,空间数据…

使用热成像仪完成地暖检测应用

家庭地暖已逐步全面走进北方家庭中,常见的地暖系统分为两类,一类为电地暖,一类为水地暖,其常见问题为地暖发热不均,地暖不热或水暖管渗漏等问题。地暖系统在保障温暖及美观的同时,一直存在维修困难的问题&a…