用队列实现栈——leetcode刷题

news2024/12/25 0:06:47

        题目的要求是用两个队列实现栈,首先我们要考虑队列的特点:先入先出,栈的特点:后入先出,所以我们的目标就是如何让先入栈的成员后出栈,后入栈的成员先出栈。

        因为有两个队列,于是我们可以这样想:让 队列1 的成员除最后一个成员出队列到 队列2,这时 队列1 剩下的成员不就是要出栈的成员吗?(即后入先出)那么我们如何控制让 队列1 出队列到只剩下一个成员呢?我们可以使用size函数,即 size(队列1) 是1的时候再Pop或者Peek,就是出栈。如图所示:


        我没有提到入栈,入栈其实就是把数据按照顺序放进 队列1 中,因为队列和栈在存储数据方面都是顺序存放的,只有在出数据的时候顺序不同,这里的 队列1 就可以理解为栈的拷贝,队列2 就可以理解为出栈的工具,因为只有在出栈时才会占用 队列2 。

        Pop完或者Peek查看完并没有结束,我们要把 队列2 中的成员再归还到 队列1 中,如果是Pop(出栈函数,删除栈顶数据),那么 队列2 直接出队列到 队列1 即可,如果是Peek(查看栈顶数据,不删除),那么 队列1 中的成员要先出队列到 队列2 在执行出队列到 队列1 的操作,如图:

接下来是代码实现,

首先还是创建栈结构体和结构体中的两个队列:


struct QueueList {
	int val;
	struct QueueList* next;
};
struct Queue {
	struct QueueList* head;
	struct QueueList* tail;
};
typedef struct {
    struct Queue* q1;
    struct Queue* q2;
} MyStack;

MyStack* myStackCreate() {
    MyStack* stack=(MyStack*)malloc(sizeof(MyStack));
    stack->q1=(struct Queue*)malloc(sizeof(struct Queue));
    stack->q2=(struct Queue*)malloc(sizeof(struct Queue));
    QueueInit(stack->q1);
    QueueInit(stack->q2);
    return stack;
}

        主要是最后两块代码,创建栈结构体并且为栈申请空间。这里我一开始犯了一个错误,没有给每个队列的指针申请空间,导致q1和q2是野指针,所以也提醒了我一次malloc是为两个指针存放开辟空间, 但并没有为指针指向的位置开辟空间,导致野指针的问题。所以要二次malloc为队列指针开辟空间。

接着是入栈函数:

void myStackPush(MyStack* obj, int x) {
    QueuePushBack(obj->q1,x);
}

        直接就是把数据放进 队列1 q1中去。

然后是Pop(出栈函数,删除栈顶)和Peek(查看栈顶成员函数):

int myStackPop(MyStack* obj) {
    while(Queue_size(obj->q1)>1){
        QueuePushBack(obj->q2,QueueFrontPop(obj->q1));
    }
    int val=QueueFrontPop(obj->q1);
    while(Queue_size(obj->q2)>0){
        QueuePushBack(obj->q1,QueueFrontPop(obj->q2));
    }
    return val;
}

int myStackTop(MyStack* obj) {
    while(Queue_size(obj->q1)>1){
        QueuePushBack(obj->q2,QueueFrontPop(obj->q1));
    }
    int val=QueueFrontPop(obj->q1);
    QueuePushBack(obj->q2,val);
    while(Queue_size(obj->q2)>0){
        QueuePushBack(obj->q1,QueueFrontPop(obj->q2));
    }
    return val;
}

        每个函数中都有两个while循环,第一个是将 队列1 q1 的成员拷贝到 队列2 q2 ,第二个是将 队列2 q2 的成员拷贝到 队列1 q1。
        先解释一下Pop函数,根据刚才的流程图,当size>1时,就是说只有一个成员时停止拷贝,这时把最后的值Pop掉并记录,然后当size>0时,就是说把 q2 的成员全部拷贝回 q1,然后返回Pop掉的值。
        Peek函数只改变了一点,就是Pop改为Peek,记录下栈顶成员,然后接着把 q1 拷贝回 q2,再整个队列拷贝回 q2。

最后是检查是否为空 Is_empty 函数和销毁内存函数:

bool myStackEmpty(MyStack* obj) {
    if(Queue_size(obj->q1)==0){
        return true;
    }else{
        return false;
    }
}

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

        因为每次调用函数过后 队列1 q1 就相当于栈,所以只需要检查 队列1 是否为空即可。

这就是文章的全部内容了,希望对你有所帮助,如有错误欢迎评论。

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

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

相关文章

[Java EE] 多线程(七): 锁策略

🌸个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 🏵️热门专栏:🍕 Collection与数据结构 (90平均质量分)https://blog.csdn.net/2301_80050796/category_12621348.html?spm1001.2014.3001.5482 🧀Java …

ZOC8 for Mac v8.08.1激活版:卓越性能的SSH客户端

在远程连接和管理的世界中,ZOC8 for Mac以其卓越的性能和丰富的功能,成为了众多专业人士的首选SSH客户端。它支持SSH1、SSH2、Telnet、Rlogin、Serial等多种协议,让您轻松连接到远程服务器。ZOC8拥有简洁直观的界面和强大的功能设置&#xff…

VMware虚拟机中ubuntu使用记录(6)—— 如何标定单目相机的内参(张正友标定法)

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、张正友相机标定法1. 工具的准备2. 标定的步骤(1) 启动相机(2) 启动标定程序(3) 标定过程的操作(5)可能的报错 3. 标定文件内容解析 前言 张正友相机标定法…

语义分割——铁路轨道数据集

引言 亲爱的读者们,您是否在寻找某个特定数据集,用于研究或项目实践?欢迎您在评论区留言,或者通过公众号私信告诉我,您想要的数据集的类型主题。小编会竭尽全力为您寻找,并在找到后第一时间与您分享。 重…

typescript 对象数组和函数

typescript 对象数组和函数 对象 在JavaScript中,对象属于非原始类型。对象也是一种符合数组类型,由若干个对象属性构成。对象属性可以是任意数据类型,比如数组,函数或者对象等。当对象属性为函数的时候,称为方法。 …

vue3--element-plus-抽屉文件上传和富文本编辑器

一、封装组件 article/components/ArticleEdit.vue <script setup> import { ref } from vue const visibleDrawer ref(false)const open (row) > {visibleDrawer.value trueconsole.log(row) }defineExpose({open }) </script><template><!-- 抽…

Java与Go: 生产者消费者模型

什么是生产者消费者模型 生产者-消费者模型&#xff08;也称为生产者-消费者问题&#xff09;是一种常见的并发编程模型&#xff0c;用于处理多线程或多进程之间的协同工作。该模型涉及两个主要角色&#xff1a;生产者和消费者&#xff0c;一个次要角色&#xff1a;缓冲区。 生…

全方位解析Node.js:从模块系统、文件操作、事件循环、异步编程、性能优化、网络编程等高级开发到后端服务架构最佳实践以及Serverless服务部署指南

Node.js是一种基于Chrome V8引擎的JavaScript运行环境&#xff0c;专为构建高性能、可扩展的网络应用而设计。其重要性在于革新了后端开发&#xff0c;通过非阻塞I/O和事件驱动模型&#xff0c;实现了轻量级、高并发处理能力。Node.js的模块化体系和活跃的npm生态极大加速了开发…

【React】React-redux多组件间的状态传递

效果&#xff08;部分完整代码在最底部&#xff09;&#xff1a; 编写 Person 组件 上面的 Count 组件&#xff0c;已经在前面几篇写过了&#xff0c;也可以直接翻到最底部看 首先我们需要在 containers 文件夹下编写 Person 组件的容器组件 首先我们需要编写 index.jsx 文件…

一种算法分类方式及其应用

在计算机科学领域&#xff0c;算法是解决问题的有效方法&#xff0c;而对算法进行分类有助于理解它们的特性、优劣以及在不同场景下的应用。常见的算法分类方法&#xff0c;包括按设计思想、问题类型、数据结构和应用领域等&#xff0c;每一类算法会对应有其典型和实际应用。 算…

连接HiveMQ代理器实现MQTT协议传输

先下载MQTTX: MQTTX: Your All-in-one MQTT Client Toolbox 使用线上免费的MQTTX BROKER:The Free Global Public MQTT Broker | Try Now | EMQ 打开MQTTX&#xff0c;创建连接&#xff0c;点击NEW SUBSCRIPTION,创建一个主题&#xff0c;这里使用test/topic,在下面Json中填写…

大语言模型教程与实践(开源)

1.简介 大语言模型&#xff08;Large Language Models, LLMs&#xff09;的兴起确实始于OpenAI在2018年发布的GPT&#xff08;Generative Pre-trained Transformer&#xff09;&#xff0c;这一开创性工作引领了自然语言处理领域的新纪元。随后&#xff0c;2022年底ChatGPT的横…

DDD:根据maven的脚手架archetype生成ddd多模块项目目录结构

随着领域驱动的兴起&#xff0c;很多人都想学习如何进行ddd的项目开发&#xff0c;那ddd的项目结构是怎么样的&#xff1f;又是如何结合SpringBoot呢&#xff1f;那么针对这个问题&#xff0c;笔者使用maven的archetype封装一个相对通用的ddd的项目目录&#xff0c;方便一键生成…

Redisson 分布式锁和同步器

系列文章目录 文章目录 系列文章目录前言前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站,这篇文章男女通用,看懂了就去分享给你的码吧。 redisson 是基于redis的扩展库,使得redis除了应用于缓存以外,还能做队列…

Excel 批量创建sheet页

参考资料 最巧妙的Excel批量创建工作表方法 一. 需求 ⏹有如下模板&#xff0c;现想根据提供的姓名&#xff0c;批量创建sheet页&#xff0c;要求每个sheet页拥有相同的模板 二. 通过透视表&#xff0c;批量创建sheet页面 ⏹如下图所示的步骤&#xff0c;创建透视表后&#…

3.3Java全栈开发前端+后端(全栈工程师进阶之路)-前端框架VUE3框架-企业级应用-Vue组合式API

为什么要使用Composition API 一个Options API实例 在前面的课程中&#xff0c;我们都是采用 Options API&#xff08;基于选项的 API &#xff09; 来写一个组件的。下面是一个实例&#xff1a; <template> Count is: {{ count }}, doubleCount is: {{ doubleCount…

brpc中http2 grpc协议解析

搭建gRPC服务 | bRPC https://blog.csdn.net/INGNIGHT/article/details/132657099 global.cpp http2_rpc_protocol.cpp ParseH2Message解析frame header信息 ParseResult H2Context::ConsumeFrameHead( 这个是固定长度的9字节帧头部&#xff0c;length是&#xff0c;3*8bit…

七. Django项目之电商购物商城 -- 退出登录

Django项目之电商购物商城 – 退出登录状态 需要开发文档和前端资料的可私聊 退出登录主要是基于Django自带的logout模块 , 该功能只有在登录是保存了用户状态才可以实现调用 一. 创建退出视图 class LogoutView(View):def get(self , request):# 删除用户数据logout(reque…

etcd源码流程---调试环境的搭建

etcd启动命令&#xff1a; name必须设置&#xff0c;否则会用default&#xff0c;集群内不同etcd实例的名字应该是唯一的&#xff0c;因为他会有一个map(name->ip)。如果initial-cluster-state设置为new&#xff0c;那么他会创建一个新的clusterid。需要在initial-cluster中…

PHP 反序列化

一、PHP 序列化 1、对象的序列化 <?php class people{public $nameGaming;private $NationLiyue;protected $Birthday12/22;public function say(){echo "老板你好呀&#xff0c;我是和记厅的镖师&#xff0c;叫我嘉明就行&#xff0c;要运货吗你&#xff1f;"…