LeeCode:129. 求根到叶子节点数字之和(中等)

题目描述

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

题解

题解1:递归

以下内容分析自
129. 求根到叶子节点数字之和【递归中隐藏着回溯】详解
递归三部曲

  1. 确定递归函数参数和返回值
    参数:二叉树的根节点。
    返回值:无返回值。
    注: 如果需要搜索整颗二叉树,那么递归函数就不要返回值,如果要搜索其中一条符合条件的路径,递归函数就需要返回值,因为遇到符合条件的路径了就要及时返回。
  2. 确定终止条件
    遇到叶子节点。
  3. 确定递归单层逻辑
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */
class Solution {
private:
    int result;//全局变量
    vector<int> path;
    int vectorToInt(const vector<int> &vec){
        int sum = 0;
        for(int i = 0; i < vec.size(); i++){
            sum = sum*10 + vec[i];
        }
        return sum;
    }
    void traversal(TreeNode *cur){
        if(!cur->left && !cur->right){
            result += vectorToInt(path);
            //return;
        }
        if(cur->left){
            path.push_back(cur->left->val);
            traversal(cur->left);// 递归
            path.pop_back();// 回溯
        }
        if(cur->right){
            path.push_back(cur->right->val);
            traversal(cur->right);
            path.pop_back();
        }
        //return;
    }
    
public:
    int sumNumbers(TreeNode* root) {
        path.clear();
        if(root == nullptr) return 0;
        path.push_back(root->val);
        traversal(root);
        return result;

    }
};

复杂度分析

  • 时间复杂度:O(n),其中 n 是二叉树的节点个数。
  • 空间复杂度:O(n)。

以下内容分析自求根到叶子节点数字之和

题解2: 深度优先搜索

深度优先搜索是很直观的做法。从根节点开始,遍历每个节点,如果遇到叶子节点,则将叶子节点对应的数字加到数字之和。如果当前节点不是叶子节点,则计算其子节点对应的数字,然后对子节点递归遍历。

class Solution {
public:
    int dfs(TreeNode* root, int prevSum) {
        if (root == nullptr) {
            return 0;
        }
        int sum = prevSum * 10 + root->val;
        if (root->left == nullptr && root->right == nullptr) {
            return sum;
        } else {
            return dfs(root->left, sum) + dfs(root->right, sum);
        }
    }
    int sumNumbers(TreeNode* root) {
        return dfs(root, 0);
    }
};

复杂度分析

  • 时间复杂度:O(n),其中 n 是二叉树的节点个数。对每个节点访问一次。
  • 空间复杂度:O(n),其中 n 是二叉树的节点个数。空间复杂度主要取决于递归调用的栈空间,递归栈的深度等于二叉树的高度,最坏情况下,二叉树的高度等于节点个数,空间复杂度为 O(n)。

题解3:广度优先搜索

  • 使用广度优先搜索,需要维护两个队列,分别存储节点和节点对应的数字。
  • 初始时,将根节点和根节点的值分别加入两个队列。每次从两个队列分别取出一个节点和一个数字,进行如下操作:
  1. 如果当前节点是叶子节点,则将该节点对应的数字加到数字之和;
  2. 如果当前节点不是叶子节点,则获得当前节点的非空子节点,并根据当前节点对应的数字和子节点的值计算子节点对应的数字,然后将子节点和子节点对应的数字分别加入两个队列。
  • 搜索结束后,即可得到所有叶子节点对应的数字之和。
class Solution {
public:
    int sumNumbers(TreeNode* root) {
        if (root == nullptr) {
            return 0;
        }
        int sum = 0;
        queue<TreeNode*> nodeQueue;
        queue<int> numQueue;
        nodeQueue.push(root);
        numQueue.push(root->val);
        while (!nodeQueue.empty()) {
            TreeNode* node = nodeQueue.front();
            int num = numQueue.front();
            nodeQueue.pop();
            numQueue.pop();
            TreeNode* left = node->left;
            TreeNode* right = node->right;
            if (left == nullptr && right == nullptr) {
                sum += num;
            } else {
                if (left != nullptr) {
                    nodeQueue.push(left);
                    numQueue.push(num * 10 + left->val);
                }
                if (right != nullptr) {
                    nodeQueue.push(right);
                    numQueue.push(num * 10 + right->val);
                }
            }
        }
        return sum;
    }
};

复杂度分析

  • 时间复杂度:O(n),其中 n 是二叉树的节点个数。对每个节点访问一次。
  • 空间复杂度:O(n),其中 n 是二叉树的节点个数。空间复杂度主要取决于队列,每个队列中的元素个数不会超过 n。

参考

129. 求根到叶子节点数字之和【递归中隐藏着回溯】详解
二叉树:递归函数究竟什么时候需要返回值,什么时候不要返回值?
求根到叶子节点数字之和
DFS和BFS两种方式解决(最好的击败了100%的用户)
Python 栈解法好难(本题第一个栈解法)【全国最菜栈方法】

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值