程序员代码面试指南题目解析(一)

        题目一:如何仅用递归函数和栈操作逆序一个栈

      题目要求:

    一个栈依次压入 1、2、3、4、5,那么从栈顶到栈底分别为5、4、3、2、1。将这个栈
转置后,从栈顶到栈底为 1、2、3、4、5,也就是实现栈中元素的逆序,但是只能用递归函数来实现,不能用其他数据结构。

      题目解析:

   根据题目要求,我们需要使用递归函数,依次将栈中的元素进行逆置出来。我们可以根据递归函数所可能产生的影响进行问题的解决。

   由于递归函数在递归的结束的时候可以进行返回,所以我们如果拿到一个数据并删除的话,在最后返回的时候会拿到最后一个数据。所进行的操作如下:

  如果想要将数组进行逆序仅仅拿到最后一个数据是完全不够的,我们还需要思考如何将一个数组进行逆序,这个思路其实很简单。我们可以再次使用一次递归操作,每次拿出最下面的数据,并保存,之后我们递归返回的时候就可以拿到最上面的数据了。我们将数据依次压入栈中就会得到一个逆序的栈。

    根据上述思路我们可以设计两个递归函数实现栈的逆序操作,实现代码如下:

#include<iostream>
#include<stack>
using namespace std;

int getAndRemoveLastElem(stack<int>& sk)
{
	int result = sk.top();
	sk.pop();
	if (sk.empty())
	{
		return result;
	}
	else
	{
		//继续进行递归删除操作
		int last = getAndRemoveLastElem(sk);
		sk.push(result);
		return last;
	}
}

//最后通过调用递归函数
void StackReverseStack(stack<int>& sk)
{
	if (sk.empty())
	{
		return;
	}
	int i = getAndRemoveLastElem(sk);
	StackReverseStack(sk);
	sk.push(i);
}

int main()
{
	stack<int> ret;
	ret.push(1);
	ret.push(2);
	ret.push(3);
	ret.push(4);
	ret.push(5);
	StackReverseStack(ret);
	//之后打印输出栈当中的内容检查逆序的状况
	while (!ret.empty())
	{
		cout << ret.top() << endl;
		ret.pop();
	}

	return 0;
}

        题目二:用栈来求解汉诺塔问题

      题目要求: 
   汉诺塔问题比较经典,这里修改一下游戏规则:现在限制不能从最左侧的塔直接移动
到最右侧,也不能从最右侧直接移动到最左侧,而是必须经过中间。求当塔有 N层的时候,需要最少移动多少步。

    题目解析:

    这道题我们可以将其认为是一个简单的模拟类的题目,首先我们来分析问题:

    题目当中要求我们所进行的移动的步数是最少的,因此不允许由连续的可逆的步骤存在。例如:上一步从左到中间进行移动,下一步就从中间到左进行移动。这样上一步移动的步骤就是完全没有效果的,不可能是最少的移动次数。

   其次,题目当中要求我们在移动的时候必须经过中间,因此我们可以判断,当两个柱子进行移动完成之后(其中一个柱子一定是中间的柱子)一定是中间的柱子和另外一个没有进行移动的柱子进行移动操作,无非就是从中间移动到目标位置,或者从目标位置移动到中间。我们只需要符合汉诺塔的规则即可,也就是小的数据必须在大的数据之上才可以进行移动。所进行的示例如下:

    我们可以设置两个变量用于作为记录两个变化的柱子的情况,之后排除中间的柱子,让另一根柱子跟中间柱子进行数据的移动即可。根据上述思路我们可以写出如下代码:

#include<iostream>
#include<stack>
#include<vector>
#include<string>
using namespace std;
//使用栈进行解决汉诺塔问题
int stackSloveHanio(int data)
{
	stack<int> left;
	while (data)
	{
		left.push(data);
		data--;
	}
	stack<int> mid;
	stack<int> right;
	//经过第一次初始化之后进行统一的数据操作
	int from = 1;
	int to = 2;     //使用1,2,3表示左中右三个柱子
	int x = left.top();
	left.pop();
	mid.push(x);
	int count = 1;
	//之后检查三个柱子的情况得到最终想要的结果
	while (!left.empty() || !mid.empty())
	{
		//当左边跟中间的柱子都不为空的时候就继续循环查找新的初始情况
		//由判断可知,我们的from和to无论怎么移动一定与中间柱子有关,
		//因此我们只需要查找中间柱子跟另外一个没有用到的柱子的移动方向即可
		int tmp = from;
		if (from == 2)
		{
			tmp = to;
		}
		if (tmp == 1)  //下一次进行的操作就需要是从另外一个柱子跟中间柱子移动数据
		{
			//判断3柱子移动到中间,还是从中间移动到3柱子
			if (!right.empty()&&(mid.empty() || right.top() < mid.top()))
			{
				//从3移动到2
				mid.push(right.top());
				right.pop();
				from = 3;
				to = 2;
			}
			else
			{
				//从2移动到3
				right.push(mid.top());
				mid.pop();
				from = 2;
				to = 3;
			}
			count++;
		}
		if (tmp == 3)
		{
			//同理可得
			if (!left.empty()&&(mid.empty() || left.top() < mid.top()))
			{
				mid.push(left.top());
				left.pop();
				from = 1;
				to = 2;
			}
			else
			{
				left.push(mid.top());
				mid.pop();
				from = 2;
				to = 1;
			}
			count++;
		}
	}
	return count;
}
int main()
{
	cout<<stackSloveHanio(3);
	return 0;
}

        题目三:生成窗口最大值数组

      题目要求:
  有一个整型数组 arr 和一个大小为 w 的窗口从数组的最左边滑到最右边,窗口每次向
右边滑一个位置。
例如,数组为[4,3,5,4.3,3,6,7],窗口大小为 3 时:
[4 3 5] 4 3 3 6 7          窗口中最大值为 5
4 [3 5 4] 3 3 6 7          窗口中最大值为 5
4 3 [5 4 3] 3 6 7          窗口中最大值为 5
4 3 5 14 3 3] 6 7         窗口中最大值为 4
4 3 5 4 [3 3 6] 7          窗口中最大值为 6
4 3 5 4 3 [3 6 7]          窗口中最大值为 7
如果数组长度为 n,窗口大小为 w,则一共产生 n-w+1 个窗口的最大值。

请实现一个函数。
•输入:整型数组 arr,窗口大小为 w.
•输出:一个长度为 n-W+1 的数组 res,res[i]表示每一种窗口状态下的最大值。
以本题为例,结果应该返回{5,5,5,4,6,7}。

     题目解析:

   这道题目实际上想要让我们进行的操作也就是根据指定的窗口依次选出窗口内元素的最大值。拿到这道题目的时候我们首先会有一个思路:直接进行模拟不久行了嘛,每次有新的数据进入窗口的时候进行一次遍历,得到最大的值,加入到数组当中。但是我们来思考一下时间复杂度:每一次遍历窗口一遍,假设窗口的大小为M,数据总共N个,所以我们总的时间复杂度为O(MN)这并不是一个很好的算法,因此我们需要进行优化。

  书中有一个很奇妙的思想:试想一下,在我们的窗口当中有新加入的元素,当新加入的元素较大的时候前面较小的元素就不可能是最大的元素,我们可以直接前面较小的数据删除。当新加入的数据较小的时候,当前面较大的数据删除之后,那么这个暂时较小的数据就有可能是最大的了。所以我们需要将数据保存。

  借助双端队列进行优化我们的代码。我们可以进行一次数组遍历,当队列为空或者新数据小于队列尾部的数据的时候可以直接将数据的下标加入到队列的尾部,当新数据大于队列尾部的数据的时候,直接将队列尾部的数据进行弹出数据。知道队列当中的数据大于新数据或者队列为空为止。所要进行的操作如下:

    需要注意的是:由于为了方便我们判断数据的加入以及输出的边界条件,因此我们需要将数组的下标作为加入队列的元素。当我们的队头元素小于我们窗口最左侧元素的下标的时候就说明数据已经失效,需要将队头数据删除,获得下一个窗口当中最大的元素。根据上述思路我们可以编写出如下的代码:

#include<iostream>
#include<vector>
#include<queue>
using namespace std;

//编写一个函数,用于查找生成窗口最大值的数组
vector<int> createMaxWindow(vector<int> ret, int windows)
{
	//我们需要借助一个栈进行解决这个问题
	deque<int> dq;
	vector<int> finEnd;
	int count = windows;
	//进行数据的遍历操作,将我们数组当中的内容的下标依次放到栈当中,如果之后进入的数据大于之前的数据
	//就直接进行入栈操作,否则需要先执行出栈操作,等到满足小于之前的元素的时候就可以执行入栈操作了
	for (int i=0;i<ret.size();i++)
	{
		while (!dq.empty()&&ret[i]>ret[dq.front()])
		{
			dq.pop_back();
		}
		dq.push_back(i);
		//之前向栈当中插入的元素还不满,不能直接进行记录最大值
		if ( count== 1)
		{
			//进行检查并返回最大值,检查栈顶的最大值是否过期
			while (i - windows >= dq.front())
			{
				dq.pop_front();    //删除过期的最大值
			}
			//找到不过期的数据直接进行记录队头元素
			finEnd.push_back(ret[dq.front()]);
		}
		else
		{
			count--;
		}
	}
	return finEnd;
}

int main()
{
	//进行测试最大生成窗口代码
	vector<int> ret({ 1,2,3,4,5,6,7,8,9 });
	vector<int> tmp=createMaxWindow(ret, 5);
	for (auto e : tmp)
	{
		cout << e << " ";
	}
	return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

移动硬盘加了PD充电口给设备供电:未来存储与供电的完美结合

添加图片注释&#xff0c;不超过 140 字&#xff08;可选&#xff09; 一、引言 随着科技的飞速发展&#xff0c;电子设备在人们的日常生活中扮演着越来越重要的角色。与此同时&#xff0c;设备间的互联互通和供电方式的便捷性也成为了用户关注的焦点。移动硬盘&#xff0c;作…

docker的centos容器使用yum报错

错误描述 学习docker过程中&#xff0c;基于 centos 镜像自定义新的镜像。拉取一个Centos镜像&#xff0c;并运行容器&#xff0c;容器安装vim&#xff0c;报错了。 报错&#xff1a;Error: Failed to download metadata for repo appstream: Cannot prepare internal mirror…

【数组算法】598. 区间加法

给你一个 m x n 的矩阵 M 和一个操作数组 op 。矩阵初始化时所有的单元格都为 0 。ops[i] [ai, bi] 意味着当所有的 0 < x < ai 和 0 < y < bi 时&#xff0c; M[x][y] 应该加 1。 在 执行完所有操作后 &#xff0c;计算并返回 矩阵中最大整数的个数 。 示例 1: …

已经安装tensorflow,仍报错No module named ‘tensorflow‘

在安装某些python虚拟环境的教程文章中&#xff0c;经常看到有评论区说安装了但是调用显示无模块&#xff0c;例如pytorch和tensorflow等等。 其实跟之前我写过的一篇文章解决方法类似&#xff0c;就是python项目中需要应用哪个虚拟环境&#xff0c;这个项目的python解释器就选…

C语言----斐波那契数列

各位看官们好&#xff0c;当我写了上一篇博客杨辉三角后&#xff0c;有一些看官叫我讲一下斐波那契数列。对于这个大家应该是有了解的。最简单的规律就是f(n)f(n-2)f(n-1)。就是当前是前两项之和&#xff0c;然后下标1和0都是1.从第三项开始计算的。那么我们知道规律&#xff0…

1.分布式-理论

目录 一、什么是分布式系统 二、CAP理论 1.一致性Consisency 2.可用性(Availability) 3.分区容错性(Partition tolrance) 三、BASE理论 1.Basically Available(基本可用) 2.Soft state&#xff08;软状态&#xff09; 3.Eventually consistent&#xff08;最终一致性&a…

QueryPerformanceCounter实现高精度uS(微妙)延时

参考连接 C# 利用Kernel32的QueryPerformanceCounter封装的 高精度定时器Timer_kernel32.dll queryperformancecounter-CSDN博客https://blog.csdn.net/wuyuander/article/details/111831973 特此记录 anlog 2024年5月11日

matlab的imclose()详解

J imclose(I,SE) J imclose(I,nhood) 说明 J imclose(I,SE) 使用结构元素 SE 对灰度或二值图像 I 执行形态学闭运算。形态学闭运算是先膨胀后腐蚀&#xff0c;这两种运算使用相同的结构元素。 J imclose(I,nhood) 对图像 I 执行闭运算&#xff0c;其中 nhood 是由指定结…

虚表,虚函数习题

6. 关于虚表说法正确的是&#xff08;d &#xff09; A&#xff1a;一个类只能有一张虚表 多重继承 B&#xff1a;基类中有虚函数&#xff0c;如果子类中没有重写基类的虚函数&#xff0c;此时子类与基类共用同一张虚表 即使子类重写了基类的虚函数&#xff0c;此时子类与…

Linux下安装mysql8.0(以rpm包安装)

前言&#xff1a;原文在我的博客网站中&#xff0c;持续更新数通、系统方面的知识&#xff0c;欢迎来访&#xff01; Linux下安装mysql8.0&#xff08;以rpm包安装&#xff09;https://myweb.myskillstree.cn/125.html 目录 1、查操作系统信息 2、下载mysql 8.0.34的rpm包 …

Python基础详解四

一&#xff0c;Json解析 字典转换为JSON&#xff1a; import jsondata [{"name":"袁震","age":20},{"name":"张三","age":21},{"name":"李四","age":22}] str json.dumps(data) …

Zotero 使用入门(笔记)

参考文献&#xff1a;Zotero入门完整教程-共27节-免费&#xff0c;李长太>&#xff0c; 仅供参考学习

【stm32-2】按键控制LED光敏传感器控制蜂鸣器

1.按键控制LED uint8_t GPIO_ReadInputDataBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin); //读取输入数据寄存器某一个端口的输入值 uint16_t GPIO_ReadInputData(GPIO_TypeDef* GPIOx); //读取整个输入数据寄存器 uint8_t GPIO_ReadOutputDataBit(GPIO_TypeDe…

REFORMER: 更高效的TRANSFORMER模型

大型Transformer模型通常在许多任务上都能达到最先进的结果&#xff0c;但是训练这些模型的成本可能会非常高昂&#xff0c;特别是在处理长序列时。我们引入了两种技术来提高Transformer的效率。首先&#xff0c;我们用一种使用局部敏感哈希的点积注意力替换了原来的点积注意力…

攻防世界-web-unseping

题目 知识点 PHP代码审计PHP序列化和反序列化PHP中魔术方法命令执行绕过方式 解读源码 <?php highlight_file(__FILE__);class ease{private $method;private $args;function __construct($method, $args) {$this->method $method;$this->args $args;}function …

【负载均衡式在线OJ项目day6】源文件路由功能及文件版题库构建

一.前言 前文讲到了OJ模块的设计思路&#xff0c;毫无疑问这是一个网络服务&#xff0c;我们先使用httplib&#xff0c;将源文件的路由功能实现&#xff0c;先把框架写好&#xff0c;后续再更改回调方法。 随后计划编写Modify模块&#xff0c;提供增删查改题库的功能(主要是查…

探索生产者/消费者模式:解决并发编程中的资源竞争

序言 在并发编程中&#xff0c;资源竞争是一个常见的问题。为了有效地管理资源并确保线程安全&#xff0c;需要采用一些有效的方法。其中之一是生产者/消费者模式&#xff0c;它是一种经典的并发设计模式&#xff0c;用于解决生产者和消费者之间的协作问题。本文将深入探讨生产…

sentinel-dashborard下发流控规则未生效排查解决

Sentinel简介 Sentinel是阿里巴巴开源的一个流量控制组件&#xff0c;它以流量为切入点&#xff0c;从流量控制、熔断降级、系统负载保护等多个维度来保护服务的稳定性。Sentinel的核心思想是&#xff1a;通过动态设置的规则&#xff0c;对进入系统的流量进行控制&#xff0c;…

【好困】磁场里的瞌睡虫:地磁暴真的会让我们感到疲倦吗?

【好困】磁场里的瞌睡虫&#xff1a;地磁暴真的会让我们感到疲倦吗&#xff1f; 写在最前面地磁暴真的会让我们感到疲倦吗&#xff1f;一探究竟地磁暴是什么&#xff1f;地磁暴如何影响人体&#xff1f;结论 &#x1f308;你好呀&#xff01;我是 是Yu欸 &#x1f30c; 2024每…

工业机器人应用实践之玻璃涂胶(篇二)

工业机器人 接上篇文章&#xff0c;浅谈一下实践应用&#xff0c;具体以玻璃涂胶为例&#xff1a; 了解工业机器人在玻璃涂胶领域的应用认识工具坐标系的标定方法掌握计时指令的应用掌握人机交互指令的应用掌握等待类指令用法&#xff08;WaitDI、WaitUnitl 等&#xff09;认…