你知道如何使用队列实现栈吗?(C语言)

 这时一道非常经典的题型,因为栈和队列的性质是相反的,队列的数据是先入先出,栈的数据是后入先出,那么怎样使用两个队列实现栈呢?

225. 用队列实现栈icon-default.png?t=N7T8https://leetcode.cn/problems/implement-stack-using-queues/

 这是题目的要求,如果使用C语言来实现的话,只能自己写一个队列了,这里我就不详细讲解了,具体实现思路在这:

http://t.csdnimg.cn/0SiCqicon-default.png?t=N7T8http://t.csdnimg.cn/0SiCq代码如下:

typedef int QDataType;
typedef struct QueueNode
{struct QueueNode* next;QDataType data;
}QNode;typedef struct Queue
{QNode* head;QNode* tail;int size;
}Que;
void QueueInit(Que* pq)
{assert(pq);pq->size = 0;pq->head = pq->tail = NULL;
}
void QueueDestroy(Que* pq)
{assert(pq);QNode* cur = pq->head;while (cur){QNode* next = cur->next;free(cur);cur = next;}pq->head = pq->tail = NULL;pq->size = 0;
}
void QueuePush(Que* pq, QDataType x)
{assert(pq);QNode* tmp = (QNode*)malloc(sizeof(QNode));if (tmp == NULL){perror("malloc fail");exit(-1);}tmp->data = x;tmp->next = NULL;if (pq->tail == NULL){pq->head = pq->tail = tmp;}else{pq->tail->next = tmp;pq->tail = tmp;}pq->size++;
}
void QueuePop(Que* pq)
{assert(pq);assert(pq->head);if (pq->head->next == NULL){free(pq->head);pq->head = pq->tail = NULL;}else{QNode* next = pq->head->next;free(pq->head);pq->head = next;}pq->size--;
}
QDataType QueueFront(Que* pq)
{assert(pq);assert(pq->head);return pq->head->data;
}
QDataType QueueBack(Que* pq)
{assert(pq);assert(pq->head);return pq->tail->data;
}
bool QueueEmpty(Que* pq)
{assert(pq);return pq->head == NULL;
}
int QueueSize(Que* pq)
{assert(pq);return pq->size;
}

实现思路:

在实现这个栈之前我们需要有一个具体思路,栈是后进先出,队列是先进后出,那么在插入上是没有区别的,在删除上就需要将对列的尾部删除,那么如何实现对列的尾部删除呢?这就需要将其中一个对列nonempty的数据导入到另一个对列empty,直到nonempty只剩一个数据,然后头删即可。

 

 删除之后将nonempty和empty互换即可,必须保证其中一个队列为空。

1.栈的定义

题目要求是使用两个队列实现栈,那么就直接在栈的定义里面包含两个队列即可。

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

2.栈的初始化

为栈malloc一块空间,在使用QueueInit实现两个队列的初始化。

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

3.数据入栈

数据入栈需要将数据push到不为空的那个队列,使用QueueEmpty判断队列是否为空,再使用QueuePush尾插数据。

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

4.数据出栈

这个是题目的难点,创建两个变量分别为nonempty(非空队列)和empty(空队列),在使用if判断q1和q2哪个为空。使用while循环来实现遍历插入和删除,结束条件为nonempty内的数据为1,也就是队列的尾部数据,在循环内使用QueuePush将nonempty的头部数据插入到empty,每次插入之后要删除掉原节点。到这里还需要注意的是,题目要求返回这个数据,所以要创建一个变量返回这个数据,最后再删除掉,始终保存一个队列为空。

int myStackPop(MyStack* obj) 
{Que* empty=&obj->q1;Que* nonempty=&obj->q2;if(!QueueEmpty(&obj->q1)){nonempty=&obj->q1;empty=&obj->q2;}else{nonempty=&obj->q2;empty=&obj->q1;}//将前size-1个元素导入空队列while(QueueSize(nonempty)>1){QueuePush(empty,QueueFront(nonempty));QueuePop(nonempty);}int ret= QueueFront(nonempty);QueuePop(nonempty);return ret;
}

5.取栈顶数据

栈顶数据也就是队列的尾部数据,使用QueueBack直接取nonempty的尾部数据即可。

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

6.判断栈是否为空

栈由两个队列组成,直接使用QueueEmpty判断两个队列是否为空即可,配合&&,必须两个都为空才返回true。

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

7.毁栈销

使用QueueDestroy销毁掉两个队列,再free掉栈的空间即可。

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

完整代码 :

typedef struct 
{Que q1;Que q2;
} MyStack;MyStack* myStackCreate() 
{MyStack* obj=(MyStack*)malloc(sizeof(MyStack));QueueInit(&obj->q1);QueueInit(&obj->q2);return obj;
}void myStackPush(MyStack* obj, int x) 
{if(!QueueEmpty(&obj->q1)){QueuePush(&obj->q1,x);}else{QueuePush(&obj->q2,x);}}int myStackPop(MyStack* obj) 
{Que* empty=&obj->q1;Que* nonempty=&obj->q2;if(!QueueEmpty(&obj->q1)){nonempty=&obj->q1;empty=&obj->q2;}else{nonempty=&obj->q2;empty=&obj->q1;}//将前size-1个元素导入空队列while(QueueSize(nonempty)>1){QueuePush(empty,QueueFront(nonempty));QueuePop(nonempty);}int ret= QueueFront(nonempty);QueuePop(nonempty);return ret;
}int myStackTop(MyStack* obj) 
{if(!QueueEmpty(&obj->q1)){return  QueueBack(&obj->q1);}else{return  QueueBack(&obj->q2);}
}bool myStackEmpty(MyStack* obj) 
{return QueueEmpty(&obj->q1)&&QueueEmpty(&obj->q2);
}void myStackFree(MyStack* obj) 
{QueueDestroy(&obj->q1);QueueDestroy(&obj->q2);free(obj);
}

今天的分享到这里就结束啦!谢谢老铁们的阅读,让我们下期再见。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PS去除图片上的文字。

问题描述:如何使用PS去除图片上的文字。 解决办法: 第一步:使用框选命令选中文字所在区域,如下图所示。 第二步,右键选择填充,选择内容填充即可。 第三步,文字去除效果如下。

Linux-基本指令(1.0)

Linux是一个非常流行的操作的知识,并提供实例帮助读者更好地理解。让我们一起来学习吧!系统,也是云计算、大数据、人工智能等领域的重要基础。学习Linux命令是Linux系统管理的基础,也是开发过程中必不可少的技能。本博客将介绍Lin…

人工智能 -- 神经网络

1、什么是神经网络 什么是人工智能?通俗来讲,就是让机器能像人一样思考。这个无需解释太多,因为通过各种科幻电影我们已经对人工智能很熟悉了。大家现在感兴趣的应该是——如何实现人工智能? 从1956年夏季首次提出“人工智能”这…

[HCIE] IPSec-VPN (IKE自动模式)

概念: IKE:因特网密钥交换 实验目标:pc1与pc2互通 步骤1:R1与R3配置默认路由 R1: ip route-static 0.0.0.0 0.0.0.0 12.1.1.2 R2: ip route-static 0.0.0.0 0.0.0.0 23.1.1.2 步骤2:配ACL…

02 RANSAC算法 及 Python 实现

文章目录 02 RANSAC算法 及 Python 实现2.1 简介2.2 算法流程2.3 RANSAC 算法实现直线拟合2.4 利用 RANSAC 算法减少 ORB 特征点误匹配 02 RANSAC算法 及 Python 实现 2.1 简介 RANSAC (Random Sample Consensus,随机抽样一致)算法的 基本假…

怎么当代课老师教学生

老师朋友们,有没有帮忙当过代课老师呢?或者,没当过的老师是不是对这种职业充满了好奇?让我来分享一下,当代课老师的日常是什么样的吧! 备课 说起备课,那可是个大工程!不过&#xff…

每日一题:LeetCode-103/107.二叉树的(层序/锯齿形层序)遍历

每日一题系列(day 04) 前言: 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 &#x1f50e…

Java 设计模式——建造者模式

目录 1.概述2.结构3.实例3.1.产品类3.2.抽象建造者类3.3.具体建造者类3.4.指挥者类3.5.测试 4.优缺点5.使用场景6.模式扩展7.创建者模式对比 1.概述 建造者模式 (Builder Pattern) 是一种创建型设计模式,用于创建复杂对象。它将对象的构建过程分离成独立的部分&…

浅谈如何成为一名优秀教师

你是不是也有一个梦想,想要成为一位优秀的教师,让孩子们如沐春风,收获满满?那么,今天就让我来给你分享一下成为优秀教师的秘诀吧! 热爱教育,点燃激情 成为优秀教师,首先要有对教育的…

install pnpm : 无法加载文件的解决办法

问题描述 我在使用pnpm的时候报错 PS D:\emss\pure-admin-backend> pnpm install pnpm : 无法加载文件 C:\Users\RD-16\AppData\Roaming\npm\pnpm.ps1。未对文件 C:\Users\RD-16\AppData\Roaming\npm\pnpm.ps1 进行数字签名。无法在当前系统上运 行该脚本。有关运行脚本和设…

揭秘:如何精准定位性能瓶颈,优化系统性能?

你好,我是小濠,目前在一家准一线互联网大厂做测试开发工程师。对于一般公司普通测试工程师来说,可能性能测试做的并不是很复杂,可能只是编写下脚本,做个压测,然后输出报告结果,瓶颈分析和调优的…

软件测试 | MySQL 主键约束详解:保障数据完整性与性能优化

📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢交流讨论:欢迎加入我们一起学习!📢资源分享:耗时200小时精选的「软件测试」资…