题目描述(简单难度)

111. Minimum Depth of Binary Tree - 图1

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

解法一 递归

104 题 有些像,当时是返回根节点到叶子节点的最大深度。记得当时的代码很简单。

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

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

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

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

  1. 3
  2. / \
  3. 9 20
  4. / \
  5. 15 7

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

但如果是下边的样子呢?

  1. 3
  2. / \
  3. 9 20
  4. / / \
  5. 8 15 7

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

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

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

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

我对代码的修正如下:

  1. public int minDepth(TreeNode root) {
  2. if (root == null) {
  3. return 0;
  4. }
  5. return minDepthHelper(root);
  6. }
  7. private int minDepthHelper(TreeNode root) {
  8. //到达叶子节点就返回 1
  9. if (root.left == null && root.right == null) {
  10. return 1;
  11. }
  12. //左孩子为空,只考虑右孩子的方向
  13. if (root.left == null) {
  14. return minDepthHelper(root.right) + 1;
  15. }
  16. //右孩子为空,只考虑左孩子的方向
  17. if (root.right == null) {
  18. return minDepthHelper(root.left) + 1;
  19. }
  20. //既有左孩子又有右孩子,那么就选一个较小的
  21. return Math.min(minDepthHelper(root.left), minDepthHelper(root.right)) + 1;
  22. }

其实也是可以把两个函数合在一起的,参考这里

  1. public int minDepth(TreeNode root) {
  2. if (root == null){
  3. return 0;
  4. }
  5. // 左孩子为空,只考虑右孩子的方向
  6. if (root.left == null) {
  7. return minDepth(root.right) + 1;
  8. }
  9. // 右孩子为空,只考虑左孩子的方向
  10. if (root.right == null) {
  11. return minDepth(root.left) + 1;
  12. }
  13. return Math.min(minDepth(root.left),minDepth(root.right)) + 1;
  14. }

此外,还有一个想法,觉得不错,大家可以看看,参考这里

  1. public int minDepth(TreeNode root) {
  2. if (root == null) {
  3. return 0;
  4. }
  5. if (root.left != null && root.right != null) {
  6. return Math.min(minDepth(root.left), minDepth(root.right)) + 1;
  7. } else {
  8. return Math.max(minDepth(root.left), minDepth(root.right)) + 1;
  9. }
  10. }

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

上边三个算法本质上其实是一样的,就是解决了一个孩子为空另一个不为空的问题,而对于104 题 没有出现这个问题,是因为我们选的是max,所以不用在意是否有一个为空。

解法二 BFS

104 题 也提供了BFS的方案,利用一个队列进行层次遍历,用一个 level 变量保存当前的深度,代码如下:

  1. public int maxDepth(TreeNode root) {
  2. Queue<TreeNode> queue = new LinkedList<TreeNode>();
  3. if (root == null)
  4. return 0;
  5. queue.offer(root);
  6. int level = 0;
  7. while (!queue.isEmpty()) {
  8. int levelNum = queue.size(); // 当前层元素的个数
  9. for (int i = 0; i < levelNum; i++) {
  10. TreeNode curNode = queue.poll();
  11. if (curNode != null) {
  12. if (curNode.left != null) {
  13. queue.offer(curNode.left);
  14. }
  15. if (curNode.right != null) {
  16. queue.offer(curNode.right);
  17. }
  18. }
  19. }
  20. level++;
  21. }
  22. return level;
  23. }

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

  1. public int minDepth(TreeNode root) {
  2. Queue<TreeNode> queue = new LinkedList<TreeNode>();
  3. if (root == null)
  4. return 0;
  5. queue.offer(root);
  6. /**********修改的地方*****************/
  7. int level = 1;
  8. /***********************************/
  9. while (!queue.isEmpty()) {
  10. int levelNum = queue.size(); // 当前层元素的个数
  11. for (int i = 0; i < levelNum; i++) {
  12. TreeNode curNode = queue.poll();
  13. if (curNode != null) {
  14. /**********修改的地方*****************/
  15. if (curNode.left == null && curNode.right == null) {
  16. return level;
  17. }
  18. /***********************************/
  19. if (curNode.left != null) {
  20. queue.offer(curNode.left);
  21. }
  22. if (curNode.right != null) {
  23. queue.offer(curNode.right);
  24. }
  25. }
  26. }
  27. level++;
  28. }
  29. return level;
  30. }

104 题 题对比着考虑的话,只要找到这道题的不同之处,代码就很好写了。

windliang wechat

添加好友一起进步~

如果觉得有帮助的话,可以点击 这里 给一个 star 哦 ^^

如果想系统的学习数据结构和算法,强烈推荐一个我之前学过的课程,可以点击 这里 查看详情