剑指 Offer 54. 二叉搜索树的第k大节点
一、题目描述
给定一棵二叉搜索树,请找出其中第 k 大的节点。
限制:
- 1 ≤ k ≤ 二叉搜索树元素个数
二、题目解析
这道题目首先得掌握以下基础概念:
- 1、二叉搜索树
- 2、中序遍历的递归写法
二叉搜索树是一棵有序的二叉树,它具有如下的性质:
- 1、若它的左子树不为空,那么左子树上的所有值均小于它的根节点
- 2、若它的右子树不为空,那么右子树上所有值均大于它的根节点
- 3、它的左子树和右子树也都是二叉搜索树
而中序遍历的递归过程是 左 —》根 —》右 的顺序。
所以,如果对二叉搜索树采取中序遍历的方式,那么得到的序列是一个从小到大排列的递增序列,而如果采取中序遍历的倒序方式,那么得到的就是一个从大到小的递减序列。
因此,求二叉搜索树的第 k 大节点就可以转换为二叉搜索树的中序遍历的倒序方式的第 k 个节点。
具体操作如下:
1、设置两个全局变量,res
用来存储最后的那个节点的值,count
用来统计记录了第几大的节点,一开始为第零大。
2、接下来对二叉搜索树采取中序遍历的倒序方式,即按照 右 —》根 —》左 的递归顺序。
3、在递归的过程中,如果发现 count == k
,说明已经找到了目标节点,返回结果就行
4、否则需要不断的执行 count++
操作,直到 count == k
位置,这样当前这个节点的值就是结果,赋值给 res
返回就行
为了帮助你更好的理解整个过程,我特意做了一组动画,点开可以查看:
三、参考代码
// 登录 AlgoMooc 官网获取更多算法图解
// https://www.algomooc.com
// 作者:程序员吴师兄
// 代码有看不懂的地方一定要私聊咨询吴师兄呀
// 剑指 Offer 54. 二叉搜索树的第k大节点:https://leetcode-cn.com/problems/er-cha-sou-suo-shu-de-di-kda-jie-dian-lcof/
class Solution {
// 设置全局变量 res,用来存储结果
int res = 0 ;
// 设置全局变量 count,用来表示记录了第几大的节点,默认是第零大
int count = 0;
public int kthLargest(TreeNode root, int k) {
// 二叉搜索树中序遍历的倒序操作,最终会生成一个递减的有序序列
helper(root,k);
// 返回第 k 大的节点值
return res;
}
void helper(TreeNode cur,int k) {
// 访问到空节点就无需再遍历了
if(cur == null) return;
// 中序遍历的倒序操作从节点的右子节点开始
helper(cur.right,k);
// 如果此时全局变量 count 已经等于 k ,说明已经获取到第 k 大的节点,无需再访问当前节点(根)的值,也无需继续遍历下去
if(count == k ) return;
// 否则更新全局变量 count 的值,并继续遍历下去
count++;
// 如果此时全局变量 count 已经等于 k ,说明已经获取到第 k 大的节点,无需再遍历下去,直接获取当前节点的值,赋值给 res
if(count == k ) res = cur.val;
// 最后再访问当前节点的左子节点
helper(cur.left,k);
}
}
2、C++ 代码
// 登录 AlgoMooc 官网获取更多算法图解
// https://www.algomooc.com
// 作者:程序员吴师兄
// 代码有看不懂的地方一定要私聊咨询吴师兄呀
// 剑指 Offer 54. 二叉搜索树的第k大节点:https://leetcode-cn.com/problems/er-cha-sou-suo-shu-de-di-kda-jie-dian-lcof/
class Solution {
// 设置全局变量 res,用来存储结果
int res = 0 ;
// 设置全局变量 count,用来表示记录了第几大的节点,默认是第零大
int count = 0;
public:
int kthLargest(TreeNode* root, int k) {
// 二叉搜索树中序遍历的倒序操作,最终会生成一个递减的有序序列
helper(root,k);
// 返回第 k 大的节点值
return res;
}
private:
void helper(TreeNode* cur,int k) {
// 访问到空节点就无需再遍历了
if(cur == NULL) return;
// 中序遍历的倒序操作从节点的右子节点开始
helper(cur->right,k);
// 如果此时全局变量 count 已经等于 k ,说明已经获取到第 k 大的节点,无需再访问当前节点(根)的值,也无需继续遍历下去
if(count == k ) return;
// 否则更新全局变量 count 的值,并继续遍历下去
count++;
// 如果此时全局变量 count 已经等于 k ,说明已经获取到第 k 大的节点,无需再遍历下去,直接获取当前节点的值,赋值给 res
if(count == k ) res = cur->val;
// 最后再访问当前节点的左子节点
helper(cur->left,k);
}
};
3、Python 代码
class Solution:
# 设置全局变量 res,用来存储结果
res = 0
# 设置全局变量 count,用来表示记录了第几大的节点,默认是第零大
count = 0
def kthLargest(self, root: TreeNode, k: int) -> int:
# 二叉搜索树中序遍历的倒序操作,最终会生成一个递减的有序序列
self.helper(root,k)
# 返回第 k 大的节点值
return self.res
def helper(self, cur: TreeNode, k: int):
# 访问到空节点就无需再遍历了
if not cur : return
# 中序遍历的倒序操作从节点的右子节点开始
self.helper(cur.right,k)
# 如果此时全局变量 count 已经等于 k ,说明已经获取到第 k 大的节点,无需再访问当前节点(根)的值,也无需继续遍历下去
if self.count == k : return
# 否则更新全局变量 count 的值,并继续遍历下去
self.count += 1
# 如果此时全局变量 count 已经等于 k ,说明已经获取到第 k 大的节点,无需再遍历下去,直接获取当前节点的值,赋值给 res
if self.count == k : self.res = cur.val
# 最后再访问当前节点的左子节点
self.helper(cur.left,k)