代码随想录Day22—part08

1. 235二叉搜索树的最近公共祖先

题目:给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先。

百度百科中最近公共祖先的定义为:“对于有根树 T 的两个结点 p、q,最近公共祖先表示为一个结点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。”

例如,给定如下二叉搜索树: root = [6,2,8,0,4,7,9,null,null,3,5]

示例 1:

  • 输入: root = [6,2,8,0,4,7,9,null,null,3,5], p = 2, q = 8

  • 输出: 6

  • 解释: 节点 2 和节点 8 的最近公共祖先是 6。

示例 2:

  • 输入: root = [6,2,8,0,4,7,9,null,null,3,5], p = 2, q = 4

  • 输出: 2

  • 解释: 节点 2 和节点 4 的最近公共祖先是 2, 因为根据定义最近公共祖先节点可以为节点本身。

说明:

  • 所有节点的值都是唯一的。

  • p、q 为不同节点且均存在于给定的二叉搜索树中。

思路:

因为是二叉搜索树,是有序树

那么只要从上到下去遍历,第一次遇到 cur 节点是数值在[p,q]区间中,则一定可以说明该节点cur就是p 和 q的最近公共祖先

解题步骤:

1.确定递归函数返回值和参数

  • 参数:当前节点,两个节点p,q

  • 返回值:返回最近公共祖先,所以是 TreeNode*

2.确定终止条件

遇到空返回就可以了

3.确定单层递归的逻辑

在遍历二叉搜索树的时候就是寻找区间[p->val, q->val]

  • 如果cur->val 大于 p->val,同时 cur->val 大于 q->val,那么应该向左遍历

  • 如果cur->val 小于 p->val,同时 cur->val 小于 q->val,那么应该向右遍历

  • 剩下的就是在区间内的情况,cur即为最近公共祖先,直接返回cur

代码如下:

#include<iostream>
#include<vector>
using namespace std;

struct TreeNode {
	int val;
	TreeNode* left;
	TreeNode* right;
	TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
};

class Solution {
private:
	TreeNode* traversal(TreeNode* cur, TreeNode* p, TreeNode* q) {
		if (cur == nullptr) return cur;

		if (cur->val > p->val && cur->val > q->val) {
			TreeNode* left = traversal(cur->left, p, q);
			if (left != nullptr) {
				return left;
			}
		}

		if (cur->val < p->val && cur->val < q->val) {
			TreeNode* right = traversal(cur->right, p, q);
			if (right != nullptr) {
				return right;
			}
		}
		return cur;
	}

public:
	TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
		return traversal(root, p, q);
	}
};

int main(void) {
	Solution solu;
	TreeNode* root = new TreeNode(6);
	root->left = new TreeNode(2);
	root->right = new TreeNode(8);
	root->left->left = new TreeNode(0);
	root->left->right = new TreeNode(4);
	root->right->left = new TreeNode(7);
	root->right->right = new TreeNode(9);
	root->left->right->left = new TreeNode(3);
	root->left->right->right = new TreeNode(5);
	TreeNode* result = solu.lowestCommonAncestor(root, root->left, root->right);
	// 输出结果
	cout << result->val << endl;
	system("pause");
	return 0;
}

2. 701二叉搜索树中的插入操作

题目:给定二叉搜索树(BST)的根节点和要插入树中的值,将值插入二叉搜索树。 返回插入后二叉搜索树的根节点。 输入数据保证,新值和原始二叉搜索树中的任意节点值都不同。

注意,可能存在多种有效的插入方式,只要树在插入后仍保持为二叉搜索树即可。 你可以返回任意有效的结果。

提示:

  • 给定的树上的节点数介于 0 和 10^4 之间

  • 每个节点都有一个唯一整数值,取值范围从 0 到 10^8

  • -10^8 <= val <= 10^8

  • 新值和原始二叉搜索树中的任意节点值都不同

思路:

可以不按照题目中提示所说的改变树的结构的插入方式。

只需要按照二叉搜索树的规则遍历,遇到空结点就插入节点就可以

 

例如插入元素10 ,需要找到末尾节点插入便可,一样的道理来插入元素15,插入元素0,插入元素6,需要调整二叉树的结构么? 并不需要。

只要遍历二叉搜索树,找到空节点 插入元素就可以了,那么这道题其实就简单了。

解题步骤:

1.确定递归函数的参数和返回值

  • 参数:根节点指针和要插入元素

  • 返回值:返回类型为节点类型 TreeNode* ,有返回值可以利用返回值完成新加入的节点与其父节点的赋值操作

TreeNode* insertIntoBST(TreeNode* root, int val)

2.确定终止条件

终止条件就是找到遍历的节点为 nullptr 的时候,就要插入节点的位置,并把插入的节点返回

if (root == nullptr) {
    TreeNode* node = new TreeNode(val);
    return node;
}

这里把添加的节点返回给上一层,就完成了父子节点的赋值操作了

3.确定单层递归的逻辑

此时要明确,需要遍历整棵树么?

别忘了这是搜索树,遍历整棵搜索树简直是对搜索树的侮辱,哈哈。

搜索树是有方向了,可以根据插入元素的数值,决定递归方向。

代码如下:

if (root->val > val) root->left = insertIntoBST(root->left, val);
if (root->val < val) root->right = insertIntoBST(root->right, val);
return root;

到这里,能感受到,如何通过递归函数返回值完成了新加入节点的父子关系赋值操作了,下一层将加入节点返回,本层用root->left或者root->right将其接住

代码如下:

if (root->val > val) root->left = insertIntoBST(root->left, val);
if (root->val < val) root->right = insertIntoBST(root->right, val);
return root;

代码如下:

代码如下:

```c++
#include<iostream>
using namespace std;

struct TreeNode {
	int val;
	TreeNode* left;
	TreeNode* right;
	TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
};

class Solution {
public:
	TreeNode* insertIntoBST(TreeNode* root, int val) {
		if (root == nullptr) {
			TreeNode* node = new TreeNode(val);
			return node; // 本层返回给root->left,下一层将当前节点返回
		}
		if (root->val > val) root->left = insertIntoBST(root->left, val);
		if (root->val < val) root->right = insertIntoBST(root->right, val);
		return root;
	}
}; 

int main(void) {
	Solution solu;
	TreeNode* root = new TreeNode(4);
	root->left = new TreeNode(2);
	root->right = new TreeNode(7);
	root->left->left = new TreeNode(1);
	root->left->right = new TreeNode(3);
	TreeNode* result = solu.insertIntoBST(root, 5);

	cout << result->val << endl;
	cout << result->left->val << " " << result->right->val << endl;
	cout << result->left->left->val << " " << result->left->right->val << " " << result->right->left->val << endl;
	system("pause");
	return 0;
}
```

​​​​​​​

3. 450删除二叉搜索树中的节点

题目:给定一个二叉搜索树的根节点 root 和一个值 key,删除二叉搜索树中的 key 对应的节点,并保证二叉搜索树的性质不变。返回二叉搜索树(有可能被更新)的根节点的引用。

一般来说,删除节点可分为两个步骤:

首先找到需要删除的节点; 如果找到了,删除它。 说明: 要求算法时间复杂度为 $O(h)$,h 为树的高度。

示例:

解题步骤:

1.确定递归函数的参数和返回值

  • 参数:当前节点的指针,关键值

  • 返回值:返回类型为节点类型 TreeNode* ,有返回值可以利用返回值删除节点

TreeNode* deleteNode(TreeNode* root, int key)

2.确定终止条件

遇到空返回,其实这也说明没找到删除的节点,遍历到空节点直接返回了

if (root == nullptr) return root;

3.确定单层递归的逻辑

这里就把二叉搜索树中删除节点遇到的情况都搞清楚。

有以下五种情况:

  • 第一种情况:没找到删除的节点,遍历到空节点直接返回了(终止条件)

  • 找到删除的节点:

    • 第二种情况:左右孩子都为空(叶子节点),直接删除节点,返回null为根节点

    • 第三种情况:删除节点的右孩子为空,左孩子不为空,删除节点,左孩子补位,返回左孩子为根节点

    • 第四种情况:删除节点的左孩子为空,右孩子不为空,删除节点,右孩子补位,返回右孩子为根节点

    • 第五种情况:左右孩子节点都不为空,则将删除节点的左子树头节点(左孩子)放到删除节点的右子树的最左面节点的左孩子上,返回删除节点右孩子为新的根节点。

 

 动画中的二叉搜索树中,删除元素7, 那么删除节点(元素7)的左孩子就是5,删除节点(元素7)的右子树的最左面节点是元素8。

将删除节点(元素7)的左孩子放到删除节点(元素7)的右子树的最左面节点(元素8)的左孩子上,就是把5为根节点的子树移到了8的左孩子的位置。

要删除的节点(元素7)的右孩子(元素9)为新的根节点。.

这样就完成删除元素7的逻辑。

整体代码如下:

#include<iostream>
using namespace std;
​
struct TreeNode {
    int val;
    TreeNode* left;
    TreeNode* right;
    TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
};
​
class Solution {
public:
    TreeNode* deleteNode(TreeNode* root, int key) {
        if (root == nullptr) return root; // 第一种情况:没有找到删除的节点,遍历到空节点直接返回
        if (root->val == key) {
            // 第二种情况:左右孩子都为空(叶子节点),直接删除节点,返回nullptr为根节点
            if (root->left == nullptr && root->right == nullptr) {
                delete root; // 内存释放
                return nullptr;
            }
            // 第三种情况:左孩子不为空,右孩子为空,删除节点,左孩子补位,返回左孩子为根节点
            else if (root->left != nullptr && root->right == nullptr) {
                auto retNode = root->left;
                delete root;
                return retNode;
            }
            // 第四种情况:右孩子不为空,左孩子为空,删除节点,右孩子补位,返回右孩子为根节点
            else if (root->left == nullptr && root->right != nullptr) {
                auto retNode = root->right;
                delete root;
                return retNode;
            }
            // 第五种情况:左右孩子都不为空,则将删除节点的左子树放到删除节点的右子树的最左面节点的左孩子的位置
            // 并返回删除节点右孩子为新的根节点
            else {
                TreeNode* cur = root->right; // 先指向删除节点的右子树根节点
                while (cur->left != nullptr) {
                    cur = cur->left;
                }
                cur->left = root->left; // 把要删除的节点(root)放在cur的左孩子的位置
                TreeNode* temp = root;  // 把root节点保存一下,下面来删除
                root = root->right;     // 返回旧root的右孩子作为新root
                delete temp;            // 释放节点内存
                return root;
            }
        }
        if (root->val > key) root->left = deleteNode(root->left, key);
        if (root->val < key) root->right = deleteNode(root->right, key);
        return root;
    }
};
​
int main(void) {
    Solution solu;
    TreeNode* root = new TreeNode(5);
    root->left = new TreeNode(3);
    root->right = new TreeNode(6);
    root->left->left = new TreeNode(2);
    root->left->right = new TreeNode(4);
    root->right->right = new TreeNode(7);
    root = solu.deleteNode(root, 3);
    // 输出结果
    cout << root->val << endl;
    cout << root->left->val << " " << root->right->val << endl;
    cout << root->left->left->val << " "  << root->right->right->val << endl;
    system("pause");
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

欢仔今天学习了嘛

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

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

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

打赏作者

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

抵扣说明:

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

余额充值