1. 首页
  2. Leetcode经典148题

leetcode-111-Minimum-Depth-of-Binary-Tree

题目描述(简单难度)

leetcode-111-Minimum-Depth-of-Binary-Tree

返回从根节点到叶子节点最小深度。

解法一 递归

public int maxDepth(TreeNode root) { if (root == null) { return 0; } return Math.max(maxDepth(root.left), maxDepth(root.right)) + 1; }

这道题是不是只要把Math.max,改成Math.min就够了。

public int minDepth(TreeNode root) {
    if (root == null) {
        return 0;
    }
    return Math.min(minDepth(root.left), minDepth(root.right)) + 1;
}

粗略的想一下,似乎很完美,比如题目给的例子

     3
    / \
   9   20
      /  \
     15   7

根据代码走一遍,root.left返回 1root.right返回 2,选较小的1,加上 1 返回结果2,完美符合结果。

但如果是下边的样子呢?

     3
    / \
   9   20
  /   /  \
 8   15   7

区别在于有一个子树的拥有一个孩子,另一个孩子为空。

这样利用上边的算法,当考虑9这个子树的时候,左孩子会返回1,由于它的右孩子为null,右孩子会返回0,选较小的0,加上 1 返回结果1给上一层。

也就是最顶层的root.left依旧得到了 1,但明显是不对的,对于左子树,应该是从 98,深度应该是 2

所以代码上需要修正这个算法,再想想题目要求是从根节点到叶节点,所以如果有一个子树的左孩子或者右孩子为null了,那就意味着这个方向不可能到达叶子节点了,所以就不要再用Min函数去判断了。

我对代码的修正如下:

public int minDepth(TreeNode root) {
    if (root == null) {
        return 0;
    }
    return minDepthHelper(root);

}

private int minDepthHelper(TreeNode root) {
    //到达叶子节点就返回 1
    if (root.left == null && root.right == null) {
        return 1;
    }
    //左孩子为空,只考虑右孩子的方向
    if (root.left == null) {
        return minDepthHelper(root.right) + 1;
    }
    //右孩子为空,只考虑左孩子的方向
    if (root.right == null) {
        return minDepthHelper(root.left) + 1;
    }
    //既有左孩子又有右孩子,那么就选一个较小的
    return Math.min(minDepthHelper(root.left), minDepthHelper(root.right)) + 1;
}

其实也是可以把两个函数合在一起的,参考public int minDepth(TreeNode root) { if (root == null){ return 0; } // 左孩子为空,只考虑右孩子的方向 if (root.left == null) { return minDepth(root.right) + 1; } // 右孩子为空,只考虑左孩子的方向 if (root.right == null) { return minDepth(root.left) + 1; } return Math.min(minDepth(root.left),minDepth(root.right)) + 1; }

此外,还有一个想法,觉得不错,大家可以看看,参考public int minDepth(TreeNode root) { if (root == null) { return 0; } if (root.left != null && root.right != null) { return Math.min(minDepth(root.left), minDepth(root.right)) + 1; } else { return Math.max(minDepth(root.left), minDepth(root.right)) + 1; } }

当左孩子为空或者右孩子为空的时候,它就直接去选一个较大深度的,因为较小深度一定是为空的那个孩子,是我们不考虑的。

上边三个算法本质上其实是一样的,就是解决了一个孩子为空另一个不为空的问题,而对于“>104 题 也提供了BFS的方案,利用一个队列进行层次遍历,用一个 level 变量保存当前的深度,代码如下:

public int maxDepth(TreeNode root) {
    Queue<TreeNode> queue = new LinkedList<TreeNode>();
    if (root == null)
        return 0;
    queue.offer(root);
    int level = 0;
    while (!queue.isEmpty()) {
        int levelNum = queue.size(); // 当前层元素的个数
        for (int i = 0; i < levelNum; i++) {
            TreeNode curNode = queue.poll();
            if (curNode != null) {
                if (curNode.left != null) {
                    queue.offer(curNode.left);
                }
                if (curNode.right != null) { 
                    queue.offer(curNode.right);
                }
            }
        }
        level++;
    }
    return level;
}

对于这道题就比较容易修改了,只要在 for 循环中判断当前是不是叶子节点,如果是的话,返回当前的 level 就可以了。此外要把level初始化改为1,因为如果只有一个根节点,它就是叶子节点,而在代码中,level 是在 for循环以后才++的,如果被提前结束的话,此时应该返回1

public int minDepth(TreeNode root) {
    Queue<TreeNode> queue = new LinkedList<TreeNode>();
    if (root == null)
        return 0;
    queue.offer(root);
    /**********修改的地方*****************/
    int level = 1;
    /***********************************/
    while (!queue.isEmpty()) {
        int levelNum = queue.size(); // 当前层元素的个数
        for (int i = 0; i < levelNum; i++) {
            TreeNode curNode = queue.poll();
            if (curNode != null) {
                /**********修改的地方*****************/
                if (curNode.left == null && curNode.right == null) {
                    return level;
                }
                /***********************************/
                if (curNode.left != null) {
                    queue.offer(curNode.left);
                }
                if (curNode.right != null) {
                    queue.offer(curNode.right);
                }
            }
        }
        level++;
    }
    return level;
}

阅读全文

看完两件小事

如果你觉得这篇文章对你挺有启发,我想请你帮我两个小忙:

  1. 关注我们的 GitHub 博客,让我们成为长期关系
  2. 把这篇文章分享给你的朋友 / 交流群,让更多的人看到,一起进步,一起成长!
  3. 关注公众号 「方志朋」,公众号后台回复「666」 免费领取我精心整理的进阶资源教程
  4. JS中文网,Javascriptc中文网是中国领先的新一代开发者社区和专业的技术媒体,一个帮助开发者成长的社区,是给开发者用的 Hacker News,技术文章由为你筛选出最优质的干货,其中包括:Android、iOS、前端、后端等方面的内容。目前已经覆盖和服务了超过 300 万开发者,你每天都可以在这里找到技术世界的头条内容。

    本文著作权归作者所有,如若转载,请注明出处

    转载请注明:文章转载自「 Java极客技术学习 」https://www.javajike.com

    标题:leetcode-111-Minimum-Depth-of-Binary-Tree

    链接:https://www.javajike.com/article/3248.html

« leetcode-110-Balanced-Binary-Tree
leetcode-112-Path-Sum»

相关推荐

QR code