【LeetCode】每日一题129:求根到叶子节点数字之和

129.求根到叶子节点数字之和

给定一个二叉树,它的每个结点都存放一个 0-9 的数字,每条从根到叶子节点的路径都代表一个数字。
例如,从根到叶子节点路径 1->2->3 代表数字 123。
计算从根到叶子节点生成的所有数字之和。
说明: 叶子节点是指没有子节点的节点。

示例 1:

  • 输入: [1,2,3]
    1
    / \
    2 3
  • 输出: 25
  • 解释:
    从根到叶子节点路径 1->2 代表数字 12.
    从根到叶子节点路径 1->3 代表数字 13.
    因此,数字总和 = 12 + 13 = 25.

示例 2:

  • 输入: [4,9,0,5,1]
    4
    / \
    9 0
    / \
    5 1
  • 输出: 1026
  • 解释:
    从根到叶子节点路径 4->9->5 代表数字 495.
    从根到叶子节点路径 4->9->1 代表数字 491.
    从根到叶子节点路径 4->0 代表数字 40.
    因此,数字总和 = 495 + 491 + 40 = 1026.

节点类如下:

public class TreeNode {
    int val;
    TreeNode left;
    TreeNode right;
    TreeNode(int x) { val = x; }
}

思考:

题目是计算「根节点」到每个「叶子节点」对应的数字之和,必然需要遍历二叉树的每个「叶子节点」,首先想到「深度优先遍历」。当然,「广度优先遍历」也能解决问题,稍微复杂一些。

「深度优先遍历」:沿着一条路一直走到底,然后从这条路尽头的节点回退到上一个节点,再从另一条路开始走到底…,不断递归重复此过程。

「广度优先遍历」:指的是从图的一个未遍历的节点出发,先遍历这个节点的相邻节点,再依次遍历每个相邻节点的相邻节点。

「叶子节点」:没有儿子的节点。

关于两种遍历的详解,可以参考此文章,讲得非常好:https://developer.51cto.com/art/202004/614590.htm

方法一:深度优先遍历

摘自官方解答:从根节点开始,遍历每个节点,如果遇到叶子节点,则将叶子节点对应的数字加到数字之和。如果当前节点不是叶子节点,则计算其子节点对应的数字,然后对子节点递归遍历。

这里用了一个巧妙的方式:当前节点的计算值 = 父节点的计算值 × 10 + 当前节点的值
对应代码中 sum = sum * 10 + root.val

public int sumNumbers(TreeNode root) {
    int sum = 0;
    return sum(root, sum);
}

public int sum(TreeNode root, int sum){
    if (root == null) {
        return 0;
    }
    sum = sum * 10 + root.val;
    if (root.left == null && root.right == null) {
        return sum;
    }
    return sum(root.left, sum) + sum(root.right, sum);
}

方法二:广度优先遍历

同样摘自官方解答:使用广度优先搜索,需要维护两个队列,分别存储节点和节点对应的数字。
初始时,将根节点和根节点的值分别加入两个队列。每次从两个队列分别取出一个节点和一个数字,进行如下操作:
1.如果当前节点是叶子节点,则将该节点对应的数字加到数字之和;
2.如果当前节点不是叶子节点,则获得当前节点的非空子节点,并根据当前节点对应的数字和子节点的值计算子节点对应的数字,然后将子节点和子节点对应的数字分别加入两个队列。

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

    Queue<Integer> numQueue = new LinkedList<>();
    Queue<TreeNode> nodeQueue = new LinkedList<>();
    // 初始化
    numQueue.offer(root.val);
    nodeQueue.offer(root);

    int sum = 0;
    while (!nodeQueue.isEmpty()) {
        TreeNode cur = nodeQueue.poll();
        int num = numQueue.poll();
        if (cur.left == null && cur.right == null) {
            sum += num;
        } else {
            if (cur.left != null) {
                nodeQueue.offer(cur.left);
                numQueue.offer(num * 10 + cur.left.val);
            }
            if (cur.right != null) {
                nodeQueue.offer(cur.right);
                numQueue.offer(num * 10 + cur.right.val);
            }
        }
    }
    return sum;
}

运行结果:深度优先遍历

在这里插入图片描述

©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页