0
点赞
收藏
分享

微信扫一扫

二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)


目录

  • ​​1、leetcode 701. 二叉搜索树中的插入操作​​
  • ​​1、题目​​
  • ​​2、递归法​​
  • ​​3、迭代法​​
  • ​​2、leetcode 450. 二叉搜索树中的插入操作​​
  • ​​1、题目​​
  • ​​2、思路+递归法​​
  • ​​3、迭代法​​
  • ​​4、删除结点的两个方法以及注意点​​
  • ​​3、leetcode 669. 修剪二叉搜索树​​
  • ​​1、题目​​
  • ​​2、思考与递归​​
  • ​​3、迭代法​​
  • ​​4、leetcode 108. 将有序数组转换为二叉搜索树​​
  • ​​1、题目​​
  • ​​2、递归思路​​

1、leetcode 701. 二叉搜索树中的插入操作

1、题目

给定二叉搜索树(BST)的根节点和要插入树中的值,将值插入二叉搜索树。 返回插入后二叉搜索树的根节点。
输入数据 保证,新值和原始二叉搜索树中的任意节点值都不同。
注意,可能存在多种有效的插入方式,只要树在插入后仍保持为二叉搜索树即可。 你可以返回任意有效的结果 。
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_剪枝
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_剪枝_02

2、递归法

递归返回值以及参数
返回值为空,因为我们进行的是插入操作,不需要知道插在哪儿。
参数:当前结点cur以及需要插入的数值val
终止条件
如果当前指针为空,返回
单层逻辑
首先,按照val的大小,遍历左右子树。
如果当前结点的值大于val,且此结点没有左孩子,则可以将val构造的结点作为其左孩子。
如果当前结点的值小于val,且此结点没有右孩子,则可以将val构造的结点作为其右孩子。
否则return;
最后在大函数里面返回root就行了。

/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
void traversal(TreeNode* cur,int val)
{
if(cur == NULL) return;
if(cur->val > val) traversal(cur->left,val);
if(cur->val < val) traversal(cur->right,val);
if(cur->val > val && cur->left == NULL)
{
//cout<<"插入左结点"<<endl;
TreeNode* ans=new TreeNode(val);
cur->left = ans;
}
if(cur->val < val && cur->right == NULL)
{
//cout<<"插入右结点"<<endl;
TreeNode* ans=new TreeNode(val);
cur->right = ans;
}
return;
}
TreeNode* insertIntoBST(TreeNode* root, int val) {
if(root == NULL)
{
TreeNode* ans=new TreeNode(val);
return ans;
}
traversal(root,val);
return root;
}
};

进行了一些修改后的代码

/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
void traversal(TreeNode* cur,int val)
{
if(cur == NULL) return;
cout<<cur->val<<endl;
if(cur->val > val)
{
traversal(cur->left,val);
if(cur->left == NULL)
{
//cout<<"插入左结点"<<endl;
TreeNode* ans=new TreeNode(val);
cur->left = ans;
}
}
if(cur->val < val)
{
traversal(cur->right,val);
if(cur->right == NULL)
{
//cout<<"插入右结点"<<endl;
TreeNode* ans=new TreeNode(val);
cur->right = ans;
}
}
return;
}
TreeNode* insertIntoBST(TreeNode* root, int val) {
if(root == NULL)
{
TreeNode* ans=new TreeNode(val);
return ans;
}
traversal(root,val);
return root;
}
};

3、迭代法

在迭代法遍历的过程中,需要记录一下当前遍历的结点的父结点,这样才能进行插入结点操作。

/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
TreeNode* insertIntoBST(TreeNode* root, int val) {
if(root == NULL)
{
TreeNode* ans=new TreeNode(val);
return ans;
}
TreeNode* cur = root;
TreeNode* parent = root;//记录上一个结点,否则不挖赋值新结点
while(cur != NULL)
{
parent = cur;
if(cur->val > val) cur = cur->left;
else cur = cur->right;
}
TreeNode* node = new TreeNode(val);
if(val < parent->val) parent->left = node; //利用parent的结点的值进行赋值
else parent->right = node;
return root;
}
};

2、leetcode 450. 二叉搜索树中的插入操作

1、题目

给定一个二叉搜索树的根节点 root 和一个值 key,删除二叉搜索树中的 key 对应的节点,并保证二叉搜索树的性质不变。返回二叉搜索树(有可能被更新)的根节点的引用。
一般来说,删除节点可分为两个步骤:
首先找到需要删除的节点;
如果找到了,删除它。
说明: 要求算法时间复杂度为 O(h),h 为树的高度。
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_结点_03

2、思路+递归法

确定递归返回值和参数
通过递归返回值来删除结点。

TreeNode* deleteNode(TreeNode* root, int key)

确定终止条件
遇到空结点返回,说明了没有找到删除的结点。

if(root == NULL) return root;

确定单层逻辑
1、没有找到删除的结点,遍历到空结点直接返回。
2、找到了删除的结点:
【1】如果左右孩子都为空,直接删除这个结点,返回NULL为根结点
【2】如果左孩子为空,右孩子不为空,删除结点,右孩子补位,返回右孩子作为根结点
【3】如果右孩子为空,左孩子不为空,删除结点,左孩子补位,返回左孩子作为根结点
【4】如果左右孩子不为空,则将删除结点的左子树的头结点放到删除结点的右子树的最左边结点的左孩子上,返回删除结点右孩子,作为新的结点
代表的是中序遍历序列的下一个节点。即比当前节点大的最小节点二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_数据结构_04

if(root->val == val)
{
if(!root->left && !root->right) return root;
else if(!root->left && root->right) return root->right;
else if(root->left && !root->right) return root->left;
else
{
TreeNode* cur = root->right;
//找到右子树的最左边结点
while(cur->left)
{
cur = cur->left;
}
cur->left = root->left;
TreeNode* tmp = root->left;
root = root->right;
delete tmp;
return root;
}
}

将新的结点返回给上一层,上一层将其作为左或者右孩子:

if(root->val > key) root->left = deleteNode(root->left,key);
if(root->val < key) root->right= deleteNode(root->right,key);
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
TreeNode* deleteNode(TreeNode* root, int key) {
if(root == NULL) return root;
if(root->val == key)
{
if(!root->left && !root->right) return NULL; //直接删除结点,返回NULL
else if(!root->left && root->right) return root->right;
else if(root->left && !root->right) return root->left;
else
{
TreeNode* cur = root->right;
//找到右子树的最左边结点
while(cur->left)
{
cur = cur->left;
}
cur->left = root->left;
TreeNode* tmp = root;
root = root->right;
delete tmp;
return root;
}
}
if(root->val > key) root->left = deleteNode(root->left,key);
if(root->val < key) root->right= deleteNode(root->right,key);
return root;
}
};

3、迭代法

/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
TreeNode* deleteOneNode(TreeNode* target)
{
if(target == NULL) return NULL;
if(target->right == NULL) return target->left;
//如果right不为空,将left转移到right左子树的最左值。
TreeNode* cur = target->right;
while(cur->left) cur = cur->left;
cur->left = target->left;
//返回target的右子树,这样target结点就被删除了
return target->right;
}
TreeNode* deleteNode(TreeNode* root, int key) {
if(root == NULL) return NULL;
TreeNode* cur = root;
TreeNode* pre = NULL; //记录cur的父结点,用来删除cur
//找到目标结点
while(cur)
{
if(cur->val == key) break;
pre = cur;
if(cur->val > key) cur = cur->left;
else cur = cur->right;
}
if(pre == NULL) return deleteOneNode(cur); //如果搜索树只有头结点
//删除右孩子还是删除左孩子,cur是父结点的左孩子还是右孩子
if(pre->left && pre->left->val == key) pre->left = deleteOneNode(cur);
if(pre->right && pre->right->val == key) pre->right = deleteOneNode(cur);
return root;
}
};

4、删除结点的两个方法以及注意点

1、移动树,找到目标结点直接对指针指向进行改变就结束了
2、覆盖值,找到目标结点先将目标结点的值和目标结点的右子树的最左边的值进行交换;交换后,继续去递归遍历树,直到再次遍历到目标结点的值,此时目标结点已经是叶子结点,左右孩子都为NULL,直接就返回NULL指针,实现了删除操作。(先覆盖值,后删除)
3、删除一个结点不能简单地置这个结点为空,我们需要修改父结点对此结点的指向!
例如我们需要将root->right覆盖掉root,不能​​​root = root->right;​​​而是应该这样写;
parent是root的父结点,child可能是左孩子也可能是右孩子,需要按照情况指定。

parent->child =  root->right;

3、leetcode 669. 修剪二叉搜索树

1、题目

给你二叉搜索树的根节点 root ,同时给定最小边界low 和最大边界 high。通过修剪二叉搜索树,使得所有节点的值在[low, high]中。修剪树不应该改变保留在树中的元素的相对结构(即,如果没有被移除,原有的父代子代关系都应当保留)。 可以证明,存在唯一的答案。
所以结果应当返回修剪好的二叉搜索树的新的根节点。注意,根节点可能会根据给定的边界发生改变。
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_结点_05
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_算法_06

2、思考与递归

因为是要遍历整棵树并且做修改,其实不需要返回值也可以,我们也可以完成修剪。但有返回值,更方便,可以通过递归函数的返回值来移除结点

如果当前结点的值小于low,要对该结点的右孩子进行再次搜索,直到找到满足区间的结点或者空结点,最后返回right结点
如果当前结点的值大于high,要对该结点的左孩子进行再次搜索,直到找到满足区间的结点或者空结点,最后返回left结点。
自上而下遍历检查,直到整棵树遍历完。
依次遍历结点左右孩子,并将返回的结果作为当前结点的左右孩子。
最后返回当前结点。
返回的结点必然是val在区间内或者是空结点。

/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
TreeNode* trimBST(TreeNode* root, int low, int high) {
if(root == NULL) return root;
if(root->val < low)
{
TreeNode* right = trimBST(root->right,low,high);
return right;
}
if(root->val > high)
{
TreeNode* left = trimBST(root->left,low,high);
return left;
}
root->left = trimBST(root->left,low,high);
root->right = trimBST(root->right,low,high);
return root;
}
};

细致观察:

root->left = trimBST(root->left,low,high);

用结点3的左孩子把下一层返回的结点0的右孩子(结点2)接住。此时结点3的右孩子就变成了结点2,将结点0从二叉树中移除了。
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_二叉树_07

错误思路记录:先得到中序遍历的结果,然后修剪结果数组,然后通过修剪后的数组构造一个二叉搜索树.但是这个思路是错误的,因为单纯用一个中序遍历数组不能构造出唯一的二叉树,随意选择一种方法会改变二叉树的结构!
如:二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_算法_08low=1,high=3,预期结果应为:
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_结点_09
但是我们这样做的结果是:
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_剪枝_10

3、迭代法

剪枝的三个步骤:
1、将root移动到[L,R]范围内
2、剪枝左子树
3、剪枝右子树
个人认为这个思路比较好理解一点,虽然繁琐了些。

class Solution {
public:
TreeNode* trimBST(TreeNode* root, int low, int high) {
if(root == NULL) return root;
//1、处理头结点,让root移动到[L,R]范围内
while(root->val < low || root->val > high)
{
if(root->val < low) root = root->right; //小于L往右走
else root = root->left; //大于R的往左走
}
TreeNode* cur = root;
//此时root已经在[L,R]范围内了,处理左孩子元素小于L的情况
while(cur)
{
//如果左孩子比low小,就用左孩子的右孩子替代左孩子
while(cur->left && cur->left->val < low)
{
cur->left = cur->left->right;
}
//一直遍历左孩子
cur = cur->left;
}
cur = root;
//此时root'已经在范围内,处理右孩子元素大于R的情况
while(cur)
{
//如果右孩子比high大,就用右孩子的左孩子替代右孩子
while(cur->right && cur->right->val > high)
{
cur->right = cur->right->left;
}
//一直遍历右孩子
cur = cur->right;
}
return root;
}
};

4、leetcode 108. 将有序数组转换为二叉搜索树

1、题目

将一个按照升序排列的有序数组,转换为一棵高度平衡二叉搜索树。
本题中,一个高度平衡二叉树是指一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过 1。
二叉搜索树的插入、删除、修剪、构造操作(leetcode701、450、669、108)_算法_11
利用数组构造二叉树的本质就是寻找分割点,分割点作为当前结点,然后递归左区间和右区间。
有序数组的分割点我们选择数组中间位置的结点。
如果数组长度为偶数,中间结点有两个,两者取其一都可以,所以也就造成了答案不是唯一的。

2、递归思路

递归返回值以及参数
返回值为结点,我们要用返回值来构造中结点的左右孩子。
参数:数组,分割数组的左边界、分割数组的右边界,这里我们定义区间是左闭右闭的。

TreeNode* traversal(vector<int>& nums,int left,int right)

终止条件
由于区间的定义为左闭右闭,所以当left>right时,就是空结点了。

if(left > right) return nullptr;

确定单层逻辑
1、取中间值:​​​int mid = left + ((right - left) / 2);​​​这个是取靠左的中间值这样可以防止right\left过大时导致的数值越界。
2、以中间位置的元素构造结点:
​​​TreeNode* root = new TreeNode(nums[mid]);​​ 3、划分区间。root->left接住下一层左区间的构造结点,root->right接住下一层的右区间构造的结点。

int mid = left + ((right - left) / 2);
TreeNode* root = new TreeNode(nums[mid]);
root->left = traversal(nums,left,mid-1);
root->right = traversal(nums,mid+1,right);

对于数组[-10,-3,0,5,9]观察遍历过程:

区间0,4
中间结点0
区间0,1
中间结点-10
区间0,-1
中间结点NULL
区间1,1
中间结点-3
区间1,0
中间结点NULL
区间2,1
中间结点NULL
区间3,4
中间结点5
区间3,2
中间结点NULL
区间4,4
中间结点9
区间4,3
中间结点NULL
区间5,4
中间结点NULL

/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode(int x) : val(x), left(NULL), right(NULL) {}
* };
*/
class Solution {
public:
TreeNode* traversal(vector<int>& nums,int left,int right)
{
cout<<"区间"<<left<<","<<right<<endl;
if(left > right)
{
cout<<"中间结点"<<"NULL"<<endl;
return nullptr;
}
int mid = left + ((right - left) / 2);
cout<<"中间结点"<<nums[mid]<<endl;
TreeNode* root = new TreeNode(nums[mid]);
root->left = traversal(nums,left,mid-1);
root->right = traversal(nums,mid+1,right);
return root;
}
TreeNode* sortedArrayToBST(vector<int>& nums) {
TreeNode* root = traversal(nums,0,nums.size()-1);
return root;
}
};


举报

相关推荐

0 条评论