如果不對遞歸有深刻的理解,本題有點難。單純移除一個節(jié)點那還不夠,要修剪!
669. 修剪二叉搜索樹
給定一個二叉搜索樹,同時給定最小邊界L 和最大邊界 R。通過修剪二叉搜索樹,使得所有節(jié)點的值在[L, R]中 (R>=L) 。你可能需要改變樹的根節(jié)點,所以結(jié)果應(yīng)當(dāng)返回修剪好的二叉搜索樹的新的根節(jié)點。
思路
相信看到這道題目大家都感覺是一道簡單題(事實上leetcode上也標(biāo)明是簡單)。
但還真的不簡單!
遞歸法
直接想法就是:遞歸處理,然后遇到root->val < low || root->val > high
的時候直接return NULL,一波修改,趕緊利落。
不難寫出如下代碼:
classSolution{
public:
TreeNode*trimBST(TreeNode*root,intlow,inthigh){
if(root==nullptr||root->valval>high)returnnullptr;
root->left=trimBST(root->left,low,high);
root->right=trimBST(root->right,low,high);
returnroot;
}
};
然而[1, 3]區(qū)間在二叉搜索樹的中可不是單純的節(jié)點3和左孩子節(jié)點0就決定的,還要考慮節(jié)點0的右子樹。
所以以上的代碼是不可行的!
從圖中可以看出需要重構(gòu)二叉樹,想想是不是本題就有點復(fù)雜了。
其實不用重構(gòu)那么復(fù)雜。
在上圖中我們發(fā)現(xiàn)節(jié)點0并不符合區(qū)間要求,那么將節(jié)點0的右孩子 節(jié)點2 直接賦給 節(jié)點3的左孩子就可以了(就是把節(jié)點0從二叉樹中移除)
理解了最關(guān)鍵部分了我們在遞歸三部曲:
- 確定遞歸函數(shù)的參數(shù)以及返回值
這里我們?yōu)槭裁葱枰祷刂的兀?/p>
因為是要遍歷整棵樹,做修改,其實不需要返回值也可以,我們也可以完成修剪(其實就是從二叉樹中移除節(jié)點)的操作。
但是有返回值,更方便,可以通過遞歸函數(shù)的返回值來移除節(jié)點。
這樣的做法在二叉樹:搜索樹中的插入操作和二叉樹:搜索樹中的刪除操作中大家已經(jīng)了解過了。
代碼如下:
TreeNode*trimBST(TreeNode*root,intlow,inthigh)
- 確定終止條件
修剪的操作并不是在終止條件上進(jìn)行的,所以就是遇到空節(jié)點返回就可以了。
if(root==nullptr)returnnullptr;
- 確定單層遞歸的邏輯
如果root(當(dāng)前節(jié)點)的元素小于low的數(shù)值,那么應(yīng)該遞歸右子樹,并返回右子樹符合條件的頭結(jié)點。
代碼如下:
if(root->valright,low,high);//尋找符合區(qū)間[low,high]的節(jié)點
returnright;
}
如果root(當(dāng)前節(jié)點)的元素大于high的,那么應(yīng)該遞歸左子樹,并返回左子樹符合條件的頭結(jié)點。
代碼如下:
if(root->val>high){
TreeNode*left=trimBST(root->left,low,high);//尋找符合區(qū)間[low,high]的節(jié)點
returnleft;
}
接下來要將下一層處理完左子樹的結(jié)果賦給root->left,處理完右子樹的結(jié)果賦給root->right。
最后返回root節(jié)點,代碼如下:
root->left=trimBST(root->left,low,high);//root->left接入符合條件的左孩子
root->right=trimBST(root->right,low,high);//root->right接入符合條件的右孩子
returnroot;
此時大家是不是還沒發(fā)現(xiàn)這多余的節(jié)點究竟是如何從二叉樹中移除的呢?
在回顧一下上面的代碼,針對下圖中二叉樹的情況:
如下代碼相當(dāng)于把節(jié)點0的右孩子(節(jié)點2)返回給上一層,
if(root->valright,low,high);//尋找符合區(qū)間[low,high]的節(jié)點
returnright;
}
然后如下代碼相當(dāng)于用節(jié)點3的左孩子 把下一層返回的 節(jié)點0的右孩子(節(jié)點2) 接住。
root->left=trimBST(root->left,low,high);
此時節(jié)點3的右孩子就變成了節(jié)點2,將節(jié)點0從二叉樹中移除了。
最后整體代碼如下:
classSolution{
public:
TreeNode*trimBST(TreeNode*root,intlow,inthigh){
if(root==nullptr)returnnullptr;
if(root->valright,low,high);//尋找符合區(qū)間[low,high]的節(jié)點
returnright;
}
if(root->val>high){
TreeNode*left=trimBST(root->left,low,high);//尋找符合區(qū)間[low,high]的節(jié)點
returnleft;
}
root->left=trimBST(root->left,low,high);//root->left接入符合條件的左孩子
root->right=trimBST(root->right,low,high);//root->right接入符合條件的右孩子
returnroot;
}
};
精簡之后代碼如下:
classSolution{
public:
TreeNode*trimBST(TreeNode*root,intlow,inthigh){
if(root==nullptr)returnnullptr;
if(root->valreturntrimBST(root->right,low,high);
if(root->val>high)returntrimBST(root->left,low,high);
root->left=trimBST(root->left,low,high);
root->right=trimBST(root->right,low,high);
returnroot;
}
};
只看代碼,其實不太好理解節(jié)點是符合移除的,這一塊大家可以自己在模擬模擬!
迭代法
因為二叉搜索樹的有序性,不需要使用棧模擬遞歸的過程。
在剪枝的時候,可以分為三步:
- 將root移動到[L, R] 范圍內(nèi),注意是左閉右閉區(qū)間
- 剪枝左子樹
- 剪枝右子樹
代碼如下:
classSolution{
public:
TreeNode*trimBST(TreeNode*root,intL,intR){
if(!root)returnnullptr;
//處理頭結(jié)點,讓root移動到[L,R]范圍內(nèi),注意是左閉右閉
while(root!=nullptr&&(root->valval>R)){
if(root->valright;//小于L往右走
elseroot=root->left;//大于R往左走
}
TreeNode*cur=root;
//此時root已經(jīng)在[L,R]范圍內(nèi),處理左孩子元素小于L的情況
while(cur!=nullptr){
while(cur->left&&cur->left->valleft=cur->left->right;
}
cur=cur->left;
}
cur=root;
//此時root已經(jīng)在[L,R]范圍內(nèi),處理右孩子大于R的情況
while(cur!=nullptr){
while(cur->right&&cur->right->val>R){
cur->right=cur->right->left;
}
cur=cur->right;
}
returnroot;
}
};
總結(jié)
修剪二叉搜索樹其實并不難,但在遞歸法中大家可看出我費(fèi)了很大的功夫來講解如何刪除節(jié)點的,這個思路其實是比較繞的。
最終的代碼倒是很簡潔。
如果不對遞歸有深刻的理解,這道題目還是有難度的!
本題我依然給出遞歸法和迭代法,初學(xué)者掌握遞歸就可以了,如果想進(jìn)一步學(xué)習(xí),就把迭代法也寫一寫。
其他語言版本
Java
classSolution{
publicTreeNodetrimBST(TreeNoderoot,intlow,inthigh){
if(root==null){
returnnull;
}
if(root.valreturntrimBST(root.right,low,high);
}
if(root.val>high){
returntrimBST(root.left,low,high);
}
//root在[low,high]范圍內(nèi)
root.left=trimBST(root.left,low,high);
root.right=trimBST(root.right,low,high);
returnroot;
}
}
Python
classSolution:
deftrimBST(self,root:TreeNode,low:int,high:int)->TreeNode:
ifnotroot:returnroot
ifroot.valreturnself.trimBST(root.right,low,high)//尋找符合區(qū)間[low,high]的節(jié)點
ifroot.val>high:
returnself.trimBST(root.left,low,high)//尋找符合區(qū)間[low,high]的節(jié)點
root.left=self.trimBST(root.left,low,high)//root->left接入符合條件的左孩子
root.right=self.trimBST(root.right,low,high)//root->right接入符合條件的右孩子
returnroot
-
源代碼
+關(guān)注
關(guān)注
96文章
2946瀏覽量
66837 -
二叉樹
+關(guān)注
關(guān)注
0文章
74瀏覽量
12355
原文標(biāo)題:修剪一棵二叉搜索樹
文章出處:【微信號:TheAlgorithm,微信公眾號:算法與數(shù)據(jù)結(jié)構(gòu)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論