题目描述
给定一个二叉树,它的每个结点都存放一个 0-9 的数字,每条从根到叶子节点的路径都代表一个数字。
例如,从根到叶子节点路径 1->2->3 代表数字 123。
计算从根到叶子节点生成的所有数字之和。
说明: 叶子节点是指没有子节点的节点。
题解
题解1:递归
以下内容分析自
129. 求根到叶子节点数字之和【递归中隐藏着回溯】详解
递归三部曲
- 确定递归函数参数和返回值
参数:二叉树的根节点。
返回值:无返回值。
注: 如果需要搜索整颗二叉树,那么递归函数就不要返回值,如果要搜索其中一条符合条件的路径,递归函数就需要返回值,因为遇到符合条件的路径了就要及时返回。 - 确定终止条件
遇到叶子节点。 - 确定递归单层逻辑
/**
* 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:广度优先搜索
- 使用广度优先搜索,需要维护两个队列,分别存储节点和节点对应的数字。
- 初始时,将根节点和根节点的值分别加入两个队列。每次从两个队列分别取出一个节点和一个数字,进行如下操作:
- 如果当前节点是叶子节点,则将该节点对应的数字加到数字之和;
- 如果当前节点不是叶子节点,则获得当前节点的非空子节点,并根据当前节点对应的数字和子节点的值计算子节点对应的数字,然后将子节点和子节点对应的数字分别加入两个队列。
- 搜索结束后,即可得到所有叶子节点对应的数字之和。
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 栈解法好难(本题第一个栈解法)【全国最菜栈方法】