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

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

      题目要求:

    一个栈依次压入 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;
}
  • 17
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料 javaj面试、知识点,用于程序员应聘学习参考,提供代码+型等资料
### 回答1: 《C/C++程序员面试指南》是由杨国祥所著的一本关于C/C++编程语言的面试指南。该书主要针对准备找工作的C/C++程序员,帮助他们提升面试技巧,以应对各种面试环节和型。 该书从面试的准备开始,包括简历的撰写、技术知识的准备和项目经验的整理等内容。作者建议程序员要准备全面、深入的技术知识,包括基础知识、常见算法、数据结构、操作系统、网络编程等方面。同时,还提醒程序员要重视项目经验的整理和总结,以便在面试中能够有条理地展示自己的实际能力。 本书还包括了大量的面试题目解析。作者通过分类整理了各种类型的面试,包括基础知识、算法、编程、系统设计等。对于每个题目,作者给出了详细的解思路和分析过程,帮助读者理解并掌握解的方法。同时,该书还提供了大量的面试经验和技巧,包括回答常见问的技巧、应对压力的方法等,都对应聘者有很大的帮助。 总的来说,《C/C++程序员面试指南》是一本较为系统的面试指南,旨在帮助C/C++程序员提升面试技巧,增强面试成功的概率。读者可以通过阅读该书,了解面试的流程、面试官的期望、常见的面试题目及解法等内容,从而更好地准备并应对面试。无论是求职者还是面试官,读完该书都能够获得一些实用的建议和策略,提高面试的效果。 ### 回答2: 《C/C++程序员面试指南—杨国祥.pdf》是一本经典的面试准备指南,作者杨国祥是一位资深的C/C++程序员,并且在书中分享了丰富的工作经验和面试技巧。 这本书主要分为两个部分。第一部分介绍了C/C++编程的基础知识,包括语言的特性、语法、数据类型、内存管理等内容。这些基础知识是面试中经常遇到的考点,掌握了这些知识可以更好地应对面试官的提问。 第二部分是关于面试的技巧和经验分享。作者从面试的准备、面试的流程、面试的常见题目等方面进行了详细的介绍。他给出了许多实用的建议,包括如何处理面试中的陷阱问、如何准备自己的简历、如何展示自己在项目中的亮点等等。这些经验和技巧能够帮助读者提高面试的表现,增加录取的机会。 总体而言,这本书是一本非常实用的面试指南,适合那些准备进行C/C++程序员面试的求职者阅读。无论是对于初级求职者还是有一定工作经验的程序员,这本书都能够提供有用的知识和技巧。通过学习这本书,读者可以更好地准备面试,提高自己的竞争力。 ### 回答3: 《C/C++程序员面试指南》是由杨国祥编写的一本面向C/C++程序员面试准备的书籍,该书主要涵盖了程序员面试的各个方面,包括面试题目、技巧和注意事项等内容。 这本书的主要特点是针对C/C++程序员面试需求进行了全面系统的整理和总结,旨在帮助读者在面试中更好地展现自己的技术能力和解决问的能力。 《C/C++程序员面试指南》的内容丰富多样,涵盖了C/C++语言基础、算法和数据结构、操作系统、网络编程、数据库等多个方面的知识点。每个知识点都给出了一些常见的面试题目,并且对每个题目都给出了详细的解答和思路分析,以便读者更好地理解和掌握。 此外,该书还分享了一些面试的技巧和经验,如对面试前的准备、面试过程中的态度和表现等方面的建议。这些技巧和经验不仅可以帮助读者提升面试的答能力,还能帮助读者更好地与面试官沟通和展示自己的优势。 总的来说,《C/C++程序员面试指南》是一本非常实用的面试准备书籍,通过系统的整理和总结,帮助程序员更好地理解面试要求和提升面试技巧,使其在面试中更有信心和竞争力。无论是初级程序员还是有一定经验的程序员,都能从中获得一些有价值的知识和经验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

阿白逆袭记

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值