《程序员面试100题.doc》由会员分享,可在线阅读,更多相关《程序员面试100题.doc(105页珍藏版)》请在三一办公上搜索。
1、 (01)把二元查找树转变成排序的双向链表 题目:输入一棵二元查找树,将该二元查找树转换成一个排序的双向链表。要求不能创建任何新的结点,只调整指针的指向。比如将二元查找树 10 / 6 14 / / 4 8 12 16转换成双向链表4=6=8=10=12=14=16。分析:本题是微软的面试题。很多与树相关的题目都是用递归的思路来解决,本题也不例外。下面我们用两种不同的递归思路来分析。思路一:当我们到达某一结点准备调整以该结点为根结点的子树时,先调整其左子树将左子树转换成一个排好序的左子链表,再调整其右子树转换右子链表。最近链接左子链表的最右结点(左子树的最大结点)、当前结点和右子链表的最左结点
2、(右子树的最小结点)。从树的根结点开始递归调整所有结点。思路二:我们可以中序遍历整棵树。按照这个方式遍历树,比较小的结点先访问。如果我们每访问一个结点,假设之前访问过的结点已经调整成一个排序双向链表,我们再把调整当前结点的指针将其链接到链表的末尾。当所有结点都访问过之后,整棵树也就转换成一个排序双向链表了。参考代码:首先我们定义二元查找树结点的数据结构如下: struct BSTreeNode / a node in the binary search tree int m_nValue; / value of nodeBSTreeNode *m_pLeft; / left child of
3、nodeBSTreeNode *m_pRight; / right child of node ;思路一对应的代码:/ Covert a sub binary-search-tree into a sorted double-linked list/ Input: pNode - the head of the sub tree/ asRight - whether pNode is the right child of its parent/ Output: if asRight is true, return the least node in the sub-tree/ else ret
4、urn the greatest node in the sub-tree/BSTreeNode* ConvertNode(BSTreeNode* pNode, bool asRight)if(!pNode)return NULL;BSTreeNode *pLeft = NULL;BSTreeNode *pRight = NULL;/ Convert the left sub-treeif(pNode-m_pLeft)pLeft = ConvertNode(pNode-m_pLeft, false);/ Connect the greatest node in the left sub-tre
5、e to the current nodeif(pLeft)pLeft-m_pRight = pNode;pNode-m_pLeft = pLeft;/ Convert the right sub-treeif(pNode-m_pRight)pRight = ConvertNode(pNode-m_pRight, true);/ Connect the least node in the right sub-tree to the current nodeif(pRight)pNode-m_pRight = pRight;pRight-m_pLeft = pNode;BSTreeNode *p
6、Temp = pNode;/ If the current node is the right child of its parent,/ return the least node in the tree whose root is the current nodeif(asRight)while(pTemp-m_pLeft)pTemp = pTemp-m_pLeft;/ If the current node is the left child of its parent,/ return the greatest node in the tree whose root is the cu
7、rrent nodeelsewhile(pTemp-m_pRight)pTemp = pTemp-m_pRight;return pTemp;/ Covert a binary search tree into a sorted double-linked list/ Input: the head of tree/ Output: the head of sorted double-linked list/BSTreeNode* Convert(BSTreeNode* pHeadOfTree)/ As we want to return the head of the sorted doub
8、le-linked list,/ we set the second parameter to be truereturn ConvertNode(pHeadOfTree, true);思路二对应的代码:/ Covert a sub binary-search-tree into a sorted double-linked list/ Input: pNode - the head of the sub tree/ pLastNodeInList - the tail of the double-linked list/void ConvertNode(BSTreeNode* pNode,
9、BSTreeNode*& pLastNodeInList)if(pNode = NULL)return;BSTreeNode *pCurrent = pNode;/ Convert the left sub-treeif (pCurrent-m_pLeft != NULL)ConvertNode(pCurrent-m_pLeft, pLastNodeInList);/ Put the current node into the double-linked listpCurrent-m_pLeft = pLastNodeInList;if(pLastNodeInList != NULL)pLas
10、tNodeInList-m_pRight = pCurrent;pLastNodeInList = pCurrent;/ Convert the right sub-treeif (pCurrent-m_pRight != NULL)ConvertNode(pCurrent-m_pRight, pLastNodeInList);/ Covert a binary search tree into a sorted double-linked list/ Input: pHeadOfTree - the head of tree/ Output: the head of sorted doubl
11、e-linked list/BSTreeNode* Convert_Solution1(BSTreeNode* pHeadOfTree)BSTreeNode *pLastNodeInList = NULL;ConvertNode(pHeadOfTree, pLastNodeInList);/ Get the head of the double-linked listBSTreeNode *pHeadOfList = pLastNodeInList;while(pHeadOfList & pHeadOfList-m_pLeft)pHeadOfList = pHeadOfList-m_pLeft
12、;return pHeadOfList;(02)设计包含min函数的栈折叠 题目:定义栈的数据结构,要求添加一个min函数,能够得到栈的最小元素。要求函数min、push以及pop的时间复杂度都是O(1)。 分析:这是去年google的一道面试题。我看到这道题目时,第一反应就是每次push一个新元素时,将栈里所有逆序元素排序。这样栈顶元素将是最小元素。但由于不能保证最后push进栈的元素最先出栈,这种思路设计的数据结构已经不是一个栈了。在栈里添加一个成员变量存放最小元素(或最小元素的位置)。每次push一个新元素进栈的时候,如果该元素比当前的最小元素还要小,则更新最小元素。乍一看这样思路挺好的
13、。但仔细一想,该思路存在一个重要的问题:如果当前最小元素被pop出去,如何才能得到下一个最小元素?因此仅仅只添加一个成员变量存放最小元素(或最小元素的位置)是不够的。我们需要一个辅助栈。每次push一个新元素的时候,同时将最小元素(或最小元素的位置。考虑到栈元素的类型可能是复杂的数据结构,用最小元素的位置将能减少空间消耗)push到辅助栈中;每次pop一个元素出栈的时候,同时pop辅助栈。参考代码:#include #include template class CStackWithMinpublic:CStackWithMin(void) virtual CStackWithMin(void
14、) T& top(void);const T& top(void) const;void push(const T& value);void pop(void);const T& min(void) const;private:Tm_data;/ theelements of stacksize_tm_minIndex;/ the indicesof minimum elements;/ get the last element of mutable stacktemplate T& CStackWithMin:top()return m_data.back();/ get the last
15、element of non-mutable stacktemplate const T& CStackWithMin:top() constreturn m_data.back();/ insert an elment at the end of stacktemplate void CStackWithMin:push(const T& value)/ append the data into the end of m_datam_data.push_back(value);/ set the index of minimum elment in m_data at the end of
16、m_minIndexif(m_minIndex.size() = 0)m_minIndex.push_back(0);elseif(value m_datam_minIndex.back()m_minIndex.push_back(m_data.size() - 1);elsem_minIndex.push_back(m_minIndex.back();/ erease the element at the end of stacktemplate void CStackWithMin:pop()/ pop m_datam_data.pop_back();/ pop m_minIndexm_m
17、inIndex.pop_back();/ get the minimum element of stacktemplate const T& CStackWithMin:min() constassert(m_data.size() 0);assert(m_minIndex.size() 0);return m_datam_minIndex.back();举个例子演示上述代码的运行过程: 步骤 数据栈 辅助栈 最小值1.push 3 3 0 32.push 4 3,4 0,0 33.push 2 3,4,2 0,0,2 24.push 1 3,4,2,1 0,0,2,3 15.pop 3,4,
18、2 0,0,2 26.pop 3,4 0,0 37.push 0 3,4,0 0,0,2 0讨论:如果思路正确,编写上述代码不是一件很难的事情。但如果能注意一些细节无疑能在面试中加分。比如我在上面的代码中做了如下的工作: 用模板类实现。如果别人的元素类型只是int类型,模板将能给面试官带来好印象; 两个版本的top函数。在很多类中,都需要提供const和非const版本的成员访问函数; min函数中assert。把代码写的尽量安全是每个软件公司对程序员的要求; 添加一些注释。注释既能提高代码的可读性,又能增加代码量,何乐而不为?总之,在面试时如果时间允许,尽量把代码写的漂亮一些。说不定代码中的
19、几个小亮点就能让自己轻松拿到心仪的Offer。(03)求子数组的最大和折叠 题目:输入一个整形数组,数组里有正数也有负数。数组中连续的一个或多个整数组成一个子数组,每个子数组都有一个和。求所有子数组的和的最大值。要求时间复杂度为O(n)。例如输入的数组为1, -2, 3, 10, -4, 7, 2, -5,和最大的子数组为3, 10, -4, 7, 2,因此输出为该子数组的和18。分析:本题最初为2005年浙江大学计算机系的考研题的最后一道程序设计题,在2006年里包括google在内的很多知名公司都把本题当作面试题。由于本题在网络中广为流传,本题也顺利成为2006年程序员面试题中经典中的经典
20、。如果不考虑时间复杂度,我们可以枚举出所有子数组并求出他们的和。不过非常遗憾的是,由于长度为n的数组有O(n2)个子数组;而且求一个长度为n的数组的和的时间复杂度为O(n)。因此这种思路的时间是O(n3)。很容易理解,当我们加上一个正数时,和会增加;当我们加上一个负数时,和会减少。如果当前得到的和是个负数,那么这个和在接下来的累加中应该抛弃并重新清零,不然的话这个负数将会减少接下来的和。基于这样的思路,我们可以写出如下代码。参考代码:/ Find the greatest sum of all sub-arrays/ Return value: if the input is valid, r
21、eturn true, otherwise return false/bool FindGreatestSumOfSubArray(int *pData, / an arrayunsigned int nLength, / the length of arrayint &nGreatestSum / the greatest sum of all sub-arrays)/ if the input is invalid, return falseif(pData = NULL) | (nLength = 0)return false;int nCurSum = nGreatestSum = 0
22、;for(unsigned int i = 0; i nLength; +i)nCurSum += pDatai;/ if the current sum is negative, discard itif(nCurSum nGreatestSum)nGreatestSum = nCurSum;/ if all data are negative, find the greatest element in the arrayif(nGreatestSum = 0)nGreatestSum = pData0;for(unsigned int i = 1; i nGreatestSum)nGrea
23、testSum = pDatai;return true;讨论:上述代码中有两点值得和大家讨论一下: 函数的返回值不是子数组和的最大值,而是一个判断输入是否有效的标志。如果函数返回值的是子数组和的最大值,那么当输入一个空指针是应该返回什么呢?返回0?那这个函数的用户怎么区分输入无效和子数组和的最大值刚好是0这两中情况呢?基于这个考虑,本人认为把子数组和的最大值以引用的方式放到参数列表中,同时让函数返回一个函数是否正常执行的标志。 输入有一类特殊情况需要特殊处理。当输入数组中所有整数都是负数时,子数组和的最大值就是数组中的最大元素。(04)在二元树中找出和为某一值的所有路径折叠 题目:输入一个整
24、数和一棵二元树。从树的根结点开始往下访问一直到叶结点所经过的所有结点形成一条路径。打印出和与输入整数相等的所有路径。例如输入整数22和如下二元树 10 / 5 12 / 4 7 则打印出两条路径:10, 12和10, 5, 7。二元树结点的数据结构定义为:struct BinaryTreeNode / a node in the binary treeint m_nValue; / value of nodeBinaryTreeNode *m_pLeft; / left child of nodeBinaryTreeNode *m_pRight; / right child of node;分
25、析:这是百度的一道笔试题,考查对树这种基本数据结构以及递归函数的理解。当访问到某一结点时,把该结点添加到路径上,并累加当前结点的值。如果当前结点为叶结点并且当前路径的和刚好等于输入的整数,则当前的路径符合要求,我们把它打印出来。如果当前结点不是叶结点,则继续访问它的子结点。当前结点访问结束后,递归函数将自动回到父结点。因此我们在函数退出之前要在路径上删除当前结点并减去当前结点的值,以确保返回父结点时路径刚好是根结点到父结点的路径。我们不难看出保存路径的数据结构实际上是一个栈结构,因为路径要与递归调用状态一致,而递归调用本质就是一个压栈和出栈的过程。参考代码:/ Find paths whose
26、 sum equal to expected sum/void FindPath(BinaryTreeNode* pTreeNode, / a node of binary treeint expectedSum, / the expected sumstd:vector&path, / a pathfrom root to current nodeint& currentSum / the sum of path)if(!pTreeNode)return;currentSum += pTreeNode-m_nValue;path.push_back(pTreeNode-m_nValue);/
27、 if the node is a leaf, and the sum is same as pre-defined,/ the path is what we want. print the pathbool isLeaf = (!pTreeNode-m_pLeft & !pTreeNode-m_pRight);if(currentSum = expectedSum & isLeaf)std:vector:iterator iter =path.begin();for(; iter != path.end(); + iter)std:cout*itert;std:coutm_pLeft)Fi
28、ndPath(pTreeNode-m_pLeft, expectedSum, path, currentSum);if(pTreeNode-m_pRight)FindPath(pTreeNode-m_pRight, expectedSum, path, currentSum);/ when we finish visiting a node and return to its parent node,/ we should delete this node from the path and/ minus the nodes value from the current sumcurrentS
29、um -= pTreeNode-m_nValue;/!I think here is no usepath.pop_back();(05)查找最小的k个元素折叠题目:输入n个整数,输出其中最小的k个。例如输入1,2,3,4,5,6,7和8这8个数字,则最小的4个数字为1,2,3和4。分析:这道题最简单的思路莫过于把输入的n个整数排序,这样排在最前面的k个数就是最小的k个数。只是这种思路的时间复杂度为O(nlogn)。我们试着寻找更快的解决思路。我们可以开辟一个长度为k的数组。每次从输入的n个整数中读入一个数。如果数组中已经插入的元素少于k个,则将读入的整数直接放到数组中。否则长度为k的数组已经
30、满了,不能再往数组里插入元素,只能替换了。如果读入的这个整数比数组中已有k个整数的最大值要小,则用读入的这个整数替换这个最大值;如果读入的整数比数组中已有k个整数的最大值还要大,则读入的这个整数不可能是最小的k个整数之一,抛弃这个整数。这种思路相当于只要排序k个整数,因此时间复杂可以降到O(n+nlogk)。通常情况下k要远小于n,所以这种办法要优于前面的思路。这是我能够想出来的最快的解决方案。不过从给面试官留下更好印象的角度出发,我们可以进一步把代码写得更漂亮一些。从上面的分析,当长度为k的数组已经满了之后,如果需要替换,每次替换的都是数组中的最大值。在常用的数据结构中,能够在O(1)时间里
31、得到最大值的数据结构为最大堆。因此我们可以用堆(heap)来代替数组。另外,自己重头开始写一个最大堆需要一定量的代码。我们现在不需要重新去发明车轮,因为前人早就发明出来了。同样,STL中的set和multiset为我们做了很好的堆的实现,我们可以拿过来用。既偷了懒,又给面试官留下熟悉STL的好印象,何乐而不为之?参考代码:#include #include #include using namespace std;typedef multisetint, greater IntHeap;/ find k least numbers in a vector/void FindKLeastNumb
32、ers(const vector& data, / a vector of dataIntHeap& leastNumbers, / k least numbers, outputunsigned int k )leastNumbers.clear();if(k = 0 | data.size() k)return;vector:const_iterator iter = data.begin();for(; iter != data.end(); + iter)/ if less than k numbers was inserted into leastNumbersif(leastNum
33、bers.size() k)leastNumbers.insert(*iter);/ leastNumbers contains k numbers and its full nowelse/ first number in leastNumbers is the greatest oneIntHeap:iterator iterFirst = leastNumbers.begin();/ if is less than the previous greatest numberif(*iter *(leastNumbers.begin()/ replace the previous great
34、est numberleastNumbers.erase(iterFirst);leastNumbers.insert(*iter);(06)判断整数序列是不是二元查找树的后序遍历结果 折叠 题目:输入一个整数数组,判断该数组是不是某二元查找树的后序遍历的结果。如果是返回true,否则返回false。 例如输入5、7、6、9、11、10、8,由于这一整数序列是如下树的后序遍历结果: 8 / 6 10 / / 5 7 9 11因此返回true。如果输入7、4、6、5,没有哪棵树的后序遍历的结果是这个序列,因此返回false。分析:这是一道trilogy的笔试题,主要考查对二元查找树的理解。在后续
35、遍历得到的序列中,最后一个元素为树的根结点。从头开始扫描这个序列,比根结点小的元素都应该位于序列的左半部分;从第一个大于跟结点开始到跟结点前面的一个元素为止,所有元素都应该大于跟结点,因为这部分元素对应的是树的右子树。根据这样的划分,把序列划分为左右两部分,我们递归地确认序列的左、右两部分是不是都是二元查找树。参考代码:using namespace std;/ Verify whether a squence of integers are the post order traversal/ of a binary search tree (BST)/ Input: squence - th
36、e squence of integers/ length - the length of squence/ Return: return ture if the squence is traversal result of a BST,/ otherwise, return false/bool verifySquenceOfBST(int squence, int length)if(squence = NULL | length = 0)return false;/ root of a BST is at the end of post order traversal squencein
37、t root = squencelength - 1;/ the nodes in left sub-tree are less than the rootint i = 0;for(; i root)break;/ the nodes in the right sub-tree are greater than the rootint j = i;for(; j length - 1; + j)if(squencej 0)left = verifySquenceOfBST(squence, i);/ verify whether the right sub-tree is a BSTbool right = true;if(i length - 1)right = verifySquenceOfBST(squence + i, length - i - 1);return (left & right);(07)翻转句子中单词的顺序折叠 题目:输入一个英文句子,翻转句子中单词的顺序,但单词内字符的顺序不变。句子中单词以空格符隔开。为简单起见,标点符号和普通字母一样处理。例如输入“I am a student.”,则输出“stude