题解 | #二叉搜索树的第k个结点#

二叉搜索树的第k个结点

http://www.nowcoder.com/practice/ef068f602dde4d28aab2b210e859150a

题目描述

给定一棵二叉搜索树,请找出其中的第k小的TreeNode结点。

考查知识点

  • 二叉搜索树(bst)
  • 树的遍历
  • 二叉搜索树中序遍历的特殊性质

分析

1.首先我们给出二叉搜索数的定义:
如果一颗二叉树,满足如下条件

  • 若其结点的左子树不空,且左子树上所有结点的值均不大于它的根结点的值。
  • 若任意结点的右子树不空,则右子树上所有结点的值均不小于它的根结点的值。
    则我们称这样的二叉数为二叉搜索树
    如下图即为一颗二叉搜索树(以下统称为bst), 对于任意节点,其左子树的所有节点一定不大于该节点,其右子树的节点一定不小于该节点
    图片说明
    2.树的遍历: 关于树的四种遍历方式我在JZ61这道题中已经介绍,这里不做赘述,这里给需要的读者附上那题的链接:
    https://blog.nowcoder.net/n/50edaca0531e4693a2db6e0575143d72
    3.二叉搜索树中序遍历的特殊性质
    关于这个性质当结论背过即可,该性质为:bst中序遍历得到的序列即为将bst上所有节点按从小到大排序的序列,可以举个例子说明其正确性:
    图片说明

解法一:使用递归实现中序遍历,因为bst的中序遍历结果即为节点的值从小到大排序的结果,所以通过一个全局变量记录当前遍历至第k个即可

  • 优点:代码简洁,实现简单
  • 缺点:递归所耗费的时间和空间比迭代实现大得多,若是多二者要求比较苛刻时要慎重考虑
    正确代码及注释如下
/*
struct TreeNode {
    int val;
    struct TreeNode *left;
    struct TreeNode *right;
    TreeNode(int x) :
            val(x), left(NULL), right(NULL) {
    }
};
*/
class Solution {
public:
    int index = 0;
    TreeNode* ans = nullptr;
    TreeNode* KthNode(TreeNode* pRoot, int k) {
        if(!pRoot) return nullptr;    //边界处理
        KthNode(pRoot->left, k);      //处理左子树
        index ++ ;
        if(index == k) ans = pRoot;   //处理当前节点
        KthNode(pRoot->right, k);     //处理右子树
        return ans;                   //返回结果
    }
};

时间复杂度:空间复杂度与系统堆栈有关,系统栈需要记住每个节点的值,所以空间复杂度为O(n)。时间复杂度应该为O(n),根据公式T(n)=2T(n/2)+1=2(2T(n/4)+1)+1=2^logn+2^(logn-1)+...+2+1 ~= n,所以时间复杂度为O(n)

方法二:通过迭代的方式实现中序遍历,依旧是使用一个变量记录遍历至第k个

  • 优点:占用时间,空间少,速度较递归实现快
  • 缺点:代码复杂,实现难度比递归写法大

正确代码及注释如下

/*
struct TreeNode {
    int val;
    struct TreeNode *left;
    struct TreeNode *right;
    TreeNode(int x) :
            val(x), left(NULL), right(NULL) {
    }
};
*/
class Solution {
public:
    TreeNode* KthNode(TreeNode* pRoot, int k) {
        if(!pRoot || k == 0) return nullptr;   //空树与边界处理
        int idx = 0;
        stack<TreeNode*> stk;          //用栈模拟中序遍历的递归与回溯过程
        while (pRoot || stk.size()) {
            while (pRoot) {        //先遍历左子树
                stk.push(pRoot);
                pRoot = pRoot->left;  
            }
            pRoot = stk.top();
            stk.pop();                        //模拟回溯
            idx ++;
            if(idx == k) return pRoot;        //再遍历当前节点
            pRoot = pRoot->right; //最后遍历右子树
        }
        return nullptr;
    }


};

时间复杂度分析:由于不管是先序遍历还是中序遍历以及后序遍历,我们都需要利用一个辅助栈来进行每个节点的存储打印,所以每个节点都要进栈和出栈,不过是根据那种遍历方式改变的是每个节点的进栈顺序,所以时间复杂度为O(n),同样空间复杂度也为O(n),n为结点数。

全部评论

相关推荐

09-30 20:49
湖南工学院 Java
SP小夜:举报了哥,你什么都没做错,全怪我那令人作呕的嫉妒和卑微的自尊心,看见你的文字我完全破防了,我直接丢盔弃甲了,看见你这图的那一秒,我满头大汗,浑身发冷,亿郁症瞬间发作了,生活仿佛没了颜色,像是被抓住尾巴的赛亚人,带着海楼石的能力者,抽离尾兽的人柱力,像是没了光的奥特曼,彻底断绝了生的希望。我几乎都快羡慕得疯了,倒在床上蒙住被子就开始抱着枕头尖叫流泪,嘴里一边喊着卧槽卧槽,一边又忍着,我边发边哭,打字的手都是抖的,后来我的手抖得越来越厉害,从心头涌起的思想、情怀和梦想,这份歆羡和悔恨交织在一起,我的笑还挂在脸上,可是眼泪一下子就掉下来了。求你了别发了,我生活再难再穷我都不会觉得难过,只有你们发这种东西的时候,我的心里像被刀割一样的痛,打着字泪水就忍不住的往下流。每天早上7点起床晚上9点睡觉,年复一年地学到现在,憧憬着一个月赚上万块的幸福生活,憧憬着美好阳光的未来。我打开了手机,看到你的图,我感到了深深的差距,我直接跳进了家门口的井里。
点赞 评论 收藏
分享
三年之期已到我的offer快到碗里来:9硕都比不上9本
点赞 评论 收藏
分享
1 收藏 评论
分享
牛客网
牛客企业服务