剑指offer --- 用两个栈实现队列的先进先出特性

news2024/12/25 0:02:42

目录

前言

一、读懂题目

二、思路分析

三、代码呈现

总结


前言

        当我们需要实现队列的先进先出特性时,可以使用栈来模拟队列的行为。本文将介绍如何使用两个栈来实现队列,并给出具体的思路和代码实现。


一、读懂题目

题目:用两个栈实现一个队列。队列的声明如下,请实现它的两个函数 appendTail 和 deleteHead,分别完成在队列尾部插入结点和在队列头部删除结点的功能。

        当我们需要利用栈来实现队列先进先出的特性时,考虑到单个栈对存入的一组数据push后再逐位pop取出,对应的序列和输入的顺序相反,那我们是否可以用两个栈,抽象类似于负负得正的手法,使得top位逐个取出得到的序列和插入时是相同的?

二、思路分析

首先基础类定义如下,我们只需要补充里面 appendTail 和 deleteHead 两函数的定义即可:

// 用两个栈实现队列
template<typename T>
class MyQueue
{
public:
	void appendTail(const T& n);
	T deleteHead();
private:
	stack<T> st1;
	stack<T> st2;
	
};

为了便于表示和说明,设定一组数据 {a, b, c, d, e} 作为测试序列。

我们看到两个栈都可存储数据,那我们不妨先将数据全部存入st1中,那么根据栈先进后出的特性,两栈中数据的存储此时应该是这样的:

既然我们想到试试“负负得正”这种方法,那要是把st1中的元素再一 一取出放入st2中,是不是相当于把序列的顺序调转再调转,这样当我们不断取 st2.top() 时,最后按照取出的序列就是原来顺序的初始序列!

我们模拟一下这个过程:

1)对st1执行两次取首压入st2中:

2)将剩余元素从st1中全部压入st2中,此时st1为空:

 3)最后每当该模拟队列执行一次 front() 操作就返回 top2 所指向的值,每次 pop() 就从 st2 取出栈顶元素(前提是st1为空),直到 st2 栈为空:

那如果在执行 front() 过程中,夹杂着新的push()指令,应该将新元素放入st1还是st2呢?如果st1不为空,要先将st1全部按照上面的规则移入st2后再插入还是先放入st1栈顶,再统一移入st2呢?

首先我们明确每个元素都需要经历“负负得正”的过程,所以首先新加元素肯定要先经过st1才能进入st2中。那是否需要st1及时清空呢?

我们不妨以 {a, b, c, d}, {e} 模拟两批次对模拟队列执行push()指令:

假定我们在每批操作结束就将该批的元素依次弹出并置入st2中,那么当需要输入上面第二组(即{e})时,st1和st2的存储情况如下:

接下来该插入第二批数据了,但是实际功能调用过程中,不可能每次单批次插入操作之间是连续的,中间可能会有删除队列中已有元素的操作,所以我们不妨在两次 push() 指令中间加一句 pop_front() 指令,那么当接着执行一次pop_front() 指令时,对于st2而言应该将 a 元素剔除。同时我们注意到 a 元素正如我们所料位于st2的栈顶,当 st2.pop() 操作执行时,取出 a 元素,符合队列先进先出的特性。此时st1为空,st2顶部元素为b,如下图:

下一步我们进行第二批插入,将元素 e 执行 push() 操作时,显然 e 需要先进st1中,所以st1中此时不为空,元素e即为st1栈顶元素:

那么此时我们要不要把st1中的所有元素(这里仅有 e )顺手植入st2中呢?

1)如果移入st2中:

现在面临一个问题:如果我们一次性取出st2中的元素,亦或是仅取st2栈顶元素执行 st2.pop() 来获得理想的队列头,却发现得到的结果并不满足我们的设计需求,两轮输入的总序列为 {a, b, c, d, e} ,而取出序列为 {a, e, ... },显然不符合先进先出的原则。

那是否意味着第二批的输入元素应该先保存在st1中,并非单批输入结束后就将其接着压入st2中,那为什么首批输入的数据就可以直接经st1后全部置入st2呢?

难道因为开始时候 st2 为空吗?那如果后续其他批次插入时,也遵循 st2 为空后再将 st1 中所有元素置于 st2 中,会不会发生这样的冲突?

不妨我们按照这样的设计思路进行测试:

首先给出部分功能的此思路代码:

1)清空 st1 容器功能代码

template<typename T>
void MyQueue<T>::appendOver()
{
	if (st2.empty())    // 只有满足st2为空才清空st1
	{
		while (!st1.empty())
		{
			st2.push(st1.top());
			st1.pop();
		}
	}
}

 2)模拟 front() , push() 和 pop() 的函数功能

template<typename T>
T& MyQueue<T>::front()
{
	if (!st1.empty())
	{
		appendOver();
	}
	assert(!st2.empty());
	
	return st2.top();
}

template<typename T>
void MyQueue<T>::appendTail(const T& n)
{
	st1.push(n);
}

template<typename T>
T MyQueue<T>::deleteHead()
{
	assert(!empty());     // 不能同时为空

	if (st2.empty())
	{
		appendOver();
	}

	T tmp_poped = st2.top();
	st2.pop();
	return tmp_poped;
}

 3)实际测试代码

void test1()
{
	MyQueue<int> mq;
	// queue.push()模拟尾插
	for (int i = 10; i > 0; i--)
	{
		mq.appendTail(i);
	}			// 1 2 3 4 5 6 7 8 9 10 右侧栈顶

	// queue.front()模拟取首元素
	// appendTail()--push()模拟入队列操作
	// deleteHead()--pop()模拟删除队列首元素
	mq.deleteHead();		// st1中:空	    st2中:1 2 3 4 5 6 7 8 9 右侧栈顶
	mq.appendTail(100);		// st1中:100	st2中:1 2 3 4 5 6 7 8 9 右侧栈顶

	// 由于st2不为空,st1中元素不发生迁移
	// 经过上面两步 st1中:100	st2中:1 2 3 4 5 6 7 8 9
	// 假设我们再进行一组类似操作
	mq.deleteHead();		// st1中:100		st2中:1 2 3 4 5 6 7 8 右侧栈顶
	mq.appendTail(55);		// st1中:55 100	    st2中:1 2 3 4 5 6 7 8 右侧栈顶

	int val = mq.deleteHead();	// st1中:55 100	st2中:1 2 3 4 5 6 7 右侧栈顶

	printf("val = %d\n", val);	// 8
	PopAndPrintMyQueue<int>(mq);
}

按理来说结果应该和各行代码后面的理想推测相同,我们看看最后两行执行结果:

可以看到和我们推测的结果完全一致,我们不仅在上面完成了一组插入删除后,额外执行了一组插入和删除操作,最后打印整个模拟队列中剩余元素时,呈现的结果和传入的顺序也相同,同样可以采用其他各种操作,统计每次取出的元素组成的序列是否满足先进先出的特性,结果终究是符合的。

三、代码呈现

下面直接给出代码:

// 用两个栈实现队列
template<typename T>
class MyQueue
{
public:
	void appendTail(const T& n);
	T deleteHead();
	T& front();
	bool empty();
private:
	void appendOver();     // 将st1中元素的全部移入st2中 --- 前提:st2不为空
	stack<T> st1;
	stack<T> st2;
	
};

template<typename T>
void MyQueue<T>::appendTail(const T& n)
{
	st1.push(n);
}

template<typename T>
void MyQueue<T>::appendOver()
{
	if (st2.empty())    // 只有满足st2为空才清空st1
	{
		while (!st1.empty())
		{
			st2.push(st1.top());
			st1.pop();
		}
	}
}

template<typename T>
bool MyQueue<T>::empty()
{
	if (st1.empty() && st2.empty())
	{
		return true;
	}
	return false;
}

template<typename T>
T MyQueue<T>::deleteHead()
{
	assert(!empty());     // 不能同时为空

	if (st2.empty())
	{
		appendOver();
	}

	T tmp_poped = st2.top();
	st2.pop();
	return tmp_poped;
}

template<typename T>
T& MyQueue<T>::front()
{
	if (st2.empty())
	{
		appendOver();
	}
	assert(!st2.empty());
	
	return st2.top();
}

template<typename T>
void PopAndPrintMyQueue(MyQueue<T>& my_q)
{
	while (!(my_q.empty()))
	{
		cout << my_q.front() << "\t";
		my_q.deleteHead();
	}
	cout << endl;
}

void test1()
{
	MyQueue<int> mq;
	// queue.push()模拟尾插
	for (int i = 10; i > 0; i--)
	{
		mq.appendTail(i);
	}			// 1 2 3 4 5 6 7 8 9 10 右侧栈顶

	// queue.front()模拟取首元素
	// appendTail()--push()模拟入队列操作
	// deleteHead()--pop()模拟删除队列首元素
	mq.deleteHead();		// st1中:空	st2中:1 2 3 4 5 6 7 8 9 右侧栈顶
	mq.appendTail(100);		// st1中:100	st2中:1 2 3 4 5 6 7 8 9 右侧栈顶

	// 由于st2不为空,st1中元素不发生迁移
	// 经过上面两步 st1中:100	st2中:1 2 3 4 5 6 7 8 9
	// 假设我们再进行一组类似操作
	mq.deleteHead();		// st1中:100		st2中:1 2 3 4 5 6 7 8 右侧栈顶
	mq.appendTail(55);		// st1中:55 100	st2中:1 2 3 4 5 6 7 8 右侧栈顶

	int val = mq.deleteHead();	// st1中:55 100	st2中:1 2 3 4 5 6 7 右侧栈顶

	printf("val = %d\n", val);	// 8
	PopAndPrintMyQueue<int>(mq);
}

值得注意的是,上面对重要功能的安全性检查中,下面两种写法其本质是相同的:

// 1.
if (st2.empty())
{
	appendOver();
}
assert(!st2.empty());

// 2.
assert(!empty());     // 不能同时为空
if (st2.empty())
{
	appendOver();
}

总结

        本文详细介绍了如何利用栈来实现队列的先进先出特性。通过使用两个栈,我们可以将插入的顺序和取出的顺序保持一致。文章讨论了具体的实现思路,并通过代码实例进行了测试。通过测试结果,我们验证了模拟队列的各种操作都满足先进先出的特性。这种使用栈实现队列的方法在实际应用中具有重要的意义。

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

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

相关文章

ClickHouse的分片和副本

1.副本 副本的目的主要是保障数据的高可用性&#xff0c;即使一台ClickHouse节点宕机&#xff0c;那么也可以从其他服务器获得相同的数据。 Data Replication | ClickHouse Docs 1.1 副本写入流程 1.2 配置步骤 &#xff08;1&#xff09;启动zookeeper集群 &#xff08;2&…

openfeign整合sentinel出现异常

版本兼容的解决办法&#xff1a;在为userClient注入feign的接口类型时&#xff0c;添加Lazy注解。 Lazy注解是Spring Framework中的一个注解&#xff0c;它通常用于标记Bean的延迟初始化。当一个Bean被标记为Lazy时&#xff0c;Spring容器在启动时不会立即初始化这个Bean&…

SpringMVC 进阶

SpringMVC 进阶 一、拦截器 SpringMVC 中 Interceptor 拦截器的主要作⽤是拦截⽤⼾的请求并进⾏相应的处理。⽐如通过它来进⾏权限验证&#xff0c;或者是来判断⽤⼾是否登陆等操作。对于 SpringMVC 拦截器的定义⽅式有两种&#xff1a; 实现接⼝&#xff1a;org.springfram…

无线物理层安全大作业

这个标题很帅 Beamforming Optimization for Physical Layer Security in MISO Wireless NetworksProblem Stateme![在这里插入图片描述](https://img-blog.csdnimg.cn/58ebb0df787c4e23b0c7be4189ebc322.png) Beamforming Optimization for Physical Layer Security in MISO W…

websocket详解

一、什么是Websocket WebSocket 是一种在单个 TCP 连接上进行 全双工 通信的协议&#xff0c;它可以让客户端和服务器之间进行实时的双向通信。 WebSocket 使用一个长连接&#xff0c;在客户端和服务器之间保持持久的连接&#xff0c;从而可以实时地发送和接收数据。 在 Web…

【Mysql学习笔记】1 - Mysql入门

一、Mysql5.7安装配置 下载后会得到zip 安装文件解压的路径最好不要有中文和空格这里我解压到 D:\hspmysql\mysql-5.7.19-winx64 目录下 【根据自己的情况来指定目录,尽量选择空间大的盘】 添加环境变量 : 电脑-属性-高级系统设置-环境变量&#xff0c;在Path 环境变量增加mysq…

js-webApi笔记1

目录 前言 Web API的概念 什么是DOM DOM树 1、查找元素 2、其他查找元素方法 3、操作元素 4、操作元素属性 5、 操作元素样式 style 6、操作自定义属性 7、 操作表单元素属性 8、事件 9、事件绑定 10、常用鼠标事件 11、定时器 12、定时器案例 前言 Web API的概念…

京东推出数据平台云海 API接口将达700个

1月16日消息&#xff0c;继上周面对企业用户发布京东电商云解决方案后&#xff0c;日前&#xff0c;京东云平台又发布了全新的数据开放平台——“云海”&#xff0c;以开放商家、商品、点击流等相关数据。 在京东主办&#xff0c;思路网协办的京东开放云服务合作伙伴2014峰会&…

leetcode刷题日志-151反转字符串中的单词

给你一个字符串 s &#xff0c;请你反转字符串中 单词 的顺序。 单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开。 返回 单词 顺序颠倒且 单词 之间用单个空格连接的结果字符串。 注意&#xff1a;输入字符串 s中可能会存在前导空格、尾随…

加密数字货币:机遇与风险并存

随着区块链技术的发展和普及&#xff0c;加密数字货币逐渐走入人们的视线。作为一种以数字形式存在的资产&#xff0c;加密数字货币具有去中心化、匿名性和安全性高等特点&#xff0c;为人们提供了一种全新的支付方式和投资选择。然而&#xff0c;加密数字货币市场也存在着较高…

实在智能携手中国电信翼支付,全球首款Agent智能体亮相2023数字科技生态大会

11月10日-13日&#xff0c;中国电信与广东省人民政府联合主办的“2023数字科技生态大会”在广州隆重举行。本届大会以“数字科技焕新启航”为主题&#xff0c;邀请众多生态合作伙伴全方位展示数字科技新成果&#xff0c;包括数字新消费、产业数字化、智能电子、人工智能大模型等…

Vite Vue3+Element Plus框架布局

App根组件&#xff1a;框架布局 <template><el-container class"layout-container-demo" style"height: 98vh"><!-- 菜单栏 --><el-aside width"200px"><el-scrollbar><!-- router:是否启用 vue-router 模式。…

springcloud整合nacos实现服务注册

Nacos是一个开源的分布式系统服务和基础设施解决方案&#xff0c;用于实现动态服务发现、配置管理和服务治理。它可以帮助开发人员和运维团队更好地管理微服务架构中的服务实例、配置信息和服务调用。 Nacos提供了服务注册与发现、动态配置管理、服务路由和负载均衡等功能&…

02-3解析BeautifulSoup

一、基本简介 BeautifulSoup简称&#xff1a;bs4什么是BeatifulSoup&#xff1f;  BeautifulSoup&#xff0c;和lxml一样&#xff0c;是一个html的解析器&#xff0c;主要功能也是解析和提取数据优缺点&#xff1f;  缺点&#xff1a;效率没有lxml的效率高  优点&#xff1…

小游戏:贪吃蛇和俄罗斯方块(Java简单版)

贪吃蛇 package z; import java.awt.Color; import java.awt.EventQueue; import java.awt.Font; import java.awt.Frame; import java.awt.Graphics; import java.awt.Image; import java.util.ArrayList; import java.util.List; import java.util.Random;import javax.swin…

畅捷通+数环通iPaaS,实现无代码集成上千款应用

01 关于畅捷通 畅捷通信息化服务专家,为用户提供在线财务软件,云进销存管理软件,移动办公软件,帮助小微企业人、财、货、客的管理,全面服务小微企业并提供社交化、个性化、服务化、小量化的生意管理支持。 企业除了畅捷通&#xff0c;还有大大小小其他的系统&#xff0c;面临着…

听GPT 讲Rust源代码--library/core/src(6)

题目来自 A Gentle Introduction To Rust[1] File: rust/library/core/src/num/dec2flt/common.rs 在Rust源代码中&#xff0c;rust/library/core/src/num/dec2flt/common.rs的作用是定义了一些用于十进制到浮点数转化的共享逻辑。以下是对该文件内容的详细介绍&#xff1a; Bi…

Linux磁盘分区快速上手(讲解详细)

一、磁盘分区 在Linux中&#xff0c;磁盘是通过分区来使用的。分区是将一个硬盘划分成几个逻辑部分来使用&#xff0c;在每个分区中可以存储不同的文件系统。因此&#xff0c;在挂载磁盘之前&#xff0c;我们需要先对磁盘进行分区。磁盘分区的过程可以通过命令行工具或图形界面…

国家药品价格查询官网-在线网站查询方法

查询药品上市价格对于个人和机构来说都是非常有必要的&#xff0c;对个人可以很好的验证该药品是否存在虚高的情况&#xff0c;对药企来说可以根据同类药品市场价格指导自产药品的定价&#xff0c;对其它机构来说了解药品价格可以帮助选择价格合理的药品供应商&#xff0c;降低…

自费出国|药学研究人员赴澳大利亚墨尔本大学访学

澳大利亚的创新药物研发在世界上一直处于领先地位&#xff0c;考虑到签证因素&#xff0c;专职药学研究的H老师将访学目标国家定位在澳大利亚。我们为其落实了墨尔本大学的职位&#xff0c;导师的研究课题与H老师的兴趣高度契合&#xff0c;最终顺利签证并如期出国。 H老师背景…