判斷一棵樹是不是另外一棵樹的子樹
轉載自:http://zhedahht.blog.163.com/blog/static/25411174201011445550396/
題目:二叉樹的結點定義如下:
struct TreeNode
{
int m_nValue;
TreeNode* m_pLeft;
TreeNode* m_pRight;
};
輸入兩棵二叉樹A和B,判斷樹B是不是A的子結構。
例如,下圖中的兩棵樹A和B,由於A中有一部分子樹的結構和B是一樣的,因此B就是A的子結構。
1 8
/ \ / \
8 7 9 2
/ \
9 2
/ \
4 7
分析:這是2010年微軟校園招聘時的一道題目。二叉樹一直是微軟面試題中經常出現的資料結構。對微軟有興趣的讀者一定要重點關注二叉樹。
回到這個題目的本身。要查詢樹A中是否存在和樹B結構一樣的子樹,我們可以分為兩步:第一步在樹A中找到和B的根結點的值一樣的結點N,第二步再判斷樹A中以N為根結點的子樹是不是包括和樹B一樣的結構。
第一步在樹A中查詢與根結點的值一樣的結點。這實際上就是樹的遍歷。對二叉樹這種資料結構熟悉的讀者自然知道我們可以用遞迴的方法去遍歷,也可以用迴圈的方法去遍歷。由於遞迴的程式碼實現比較簡潔,面試時如果沒有特別要求,我們通常都會採用遞迴的方式。下面是參考程式碼:
bool HasSubtree(TreeNode* pTreeHead1, TreeNode* pTreeHead2)
{
if((pTreeHead1 == NULL && pTreeHead2 != NULL) ||
(pTreeHead1 != NULL && pTreeHead2 == NULL))
return false;
if(pTreeHead1 == NULL && pTreeHead2 == NULL)
return true;
return HasSubtreeCore(pTreeHead1, pTreeHead2);
}
bool HasSubtreeCore(TreeNode* pTreeHead1, TreeNode* pTreeHead2)
{
bool result = false;
if(pTreeHead1->m_nValue == pTreeHead2->m_nValue)
{
result = DoesTree1HaveAllNodesOfTree2(pTreeHead1, pTreeHead2);
}
if(!result && pTreeHead1->m_pLeft != NULL)
result = HasSubtreeCore(pTreeHead1->m_pLeft, pTreeHead2);
if(!result && pTreeHead1->m_pRight != NULL)
result = HasSubtreeCore(pTreeHead1->m_pRight, pTreeHead2);
return result;
}
在上述程式碼中,我們遞迴呼叫hasSubtreeCore遍歷二叉樹A。如果發現某一結點的值和樹B的頭結點的值相同,則呼叫DoesTree1HaveAllNodeOfTree2,做第二步判斷。
在面試的時候,我們一定要注意邊界條件的檢查,即檢查空指標。當樹A或樹B為空的時候,定義相應的輸出。如果沒有檢查並做相應的處理,程式非常容易崩潰,這是面試時非常忌諱的事情。由於沒有必要在每一次遞迴中做邊界檢查(每一次遞迴都做檢查,增加了不必要的時間開銷),上述程式碼只在HasSubtree中作了邊界檢查後,在HasSubtreeCore中作遞迴遍歷。
接下來考慮第二步,判斷以樹A中以N為根結點的子樹是不是和樹B具有相同的結構。同樣,我們也可以用遞迴的思路來考慮:如果結點N的值和樹B的根結點不相同,則以N為根結點的子樹和樹B肯定不具有相同的結點;如果他們的值相同,則遞迴地判斷他們的各自的左右結點的值是不是相同。遞迴的終止條件是我們到達了樹A或者樹B的葉結點。參考程式碼如下:
bool DoesTree1HaveAllNodesOfTree2(TreeNode* pTreeHead1, TreeNode* pTreeHead2)
{
if(pTreeHead2 == NULL)
return true;
if(pTreeHead1 == NULL)
return false;
if(pTreeHead1->m_nValue != pTreeHead2->m_nValue)
return false;
return DoesTree1HaveAllNodesOfTree2(pTreeHead1->m_pLeft, pTreeHead2->m_pLeft) &&
DoesTree1HaveAllNodesOfTree2(pTreeHead1->m_pRight, pTreeHead2->m_pRight);
}