路径总和

力扣题目链接(opens new window)

给定一个二叉树和一个目标和,判断该树中是否存在根节点到叶子节点的路径,这条路径上所有节点值相加等于目标和。

说明: 叶子节点是指没有子节点的节点。

示例: 给定如下二叉树,以及目标和 sum = 22,

返回 true, 因为存在目标和为 22 的根节点到叶子节点的路径 5->4->11->2

思路

用递归的方法做,传入target值,没遍历到一个节点就与其做差

如果最后遍历到叶子节点时target值变成0,那么就说明本条路径的节点之和满足target值

递归法

1、确定递归函数参数和返回值

输入肯定是根节点了,因为还需要把target传入,所以还需要一个计数变量int count

那么,返回值如何确定?或者说需不需要返回值?

这里引申一个问题:递归函数什么时候需要返回值?

可以总结为如下三点:

  • 如果需要搜索整棵二叉树且不用处理递归返回值,递归函数就不要返回值。(113.路径总和ii)
  • 如果需要搜索整棵二叉树且需要处理递归返回值,递归函数就需要返回值。 (236. 二叉树的最近公共祖先 (opens new window)
  • 如果要搜索其中一条符合条件的路径,那么递归一定需要返回值,因为遇到符合条件的路径了就要及时返回。(本题的情况)

因为题目只要求我们找出一条符合路径值等于target的路径,所以我们有可能不需要完整遍历整颗二叉树。且一旦找到满足条件的路径,需要及时返回

因此递归函数的返回值应该是bool

bool traversal(TreeNode* node, int count){

}

2、确定终止条件

这里要考虑如何统计全部路径节点的值

注意,不要使用累加的方式(即遍历一个节点加一个,然后最后再判断是否满足target),这种方式虽然很容易想到,但是代码实现比较麻烦

因此,我们将count设置为target的大小,让其在递归遍历的过程中递减

那么结束的条件就有以下两种:

1、遇到叶子节点(遍历结束),且当前count已经递减至0

2、遇到叶子节点(遍历结束)但count不为0

bool traversal(TreeNode* node, int count){
if(node->left == NULL && node->right == NULL && count == 0) return true;
if(node->left == NULL && node->right == NULL) return false;
}

3、确定单层处理逻辑

调用递归函数,当我们找到满足条件的路径或者已经到达叶子节点后,需要通过回溯将count值再加回原来的初值状态(等于target)

因为递归函数设置了返回值,如果我们找到符合条件的路径,就可以立刻返回true

没找到就返回false

bool traversal(TreeNode* node, int count){
//确定终止条件
//到叶子节点并找到路径
if(node->left == NULL && node->right == NULL && count == 0) return true;
//到叶子节点但没找到路径
if(node->left == NULL && node->right == NULL) return false; //确定单层处理逻辑
//调用递归函数
if(node->left){
count -= node->left->val;//不断递减
//如果递归函数返回true,则找到路径,可以直接返回true
if(traversal(node->left, count)) return true;
count += node->left->val;//回溯,撤销处理结果
}
if(node->right){
count -= node->right->val;//不断递减
//如果递归函数返回true,则找到路径,可以直接返回true
if(traversal(node->right, count)) return true;
count += node->right->val;//回溯,撤销处理结果
}
return false;//没找到返回false
}
代码
class Solution {
public:
//使用递归,遍历出所有路径,将路径之和与target比较
//确定递归函数的参数和返回值
bool traversal(TreeNode* node, int count){
//确定终止条件
//到叶子节点并找到路径
if(node->left == NULL && node->right == NULL && count == 0) return true;
//到叶子节点但没找到路径
if(node->left == NULL && node->right == NULL) return false; //确定单层处理逻辑
//调用递归函数
if(node->left){
count -= node->left->val;//不断递减
//如果递归函数返回true,则找到路径,可以直接返回true
if(traversal(node->left, count)) return true;
count += node->left->val;//回溯,撤销处理结果
}
if(node->right){
count -= node->right->val;//不断递减
//如果递归函数返回true,则找到路径,可以直接返回true
if(traversal(node->right, count)) return true;
count += node->right->val;//回溯,撤销处理结果
}
return false;//没找到返回false }
bool hasPathSum(TreeNode* root, int targetSum) {
if (root == NULL) return false;//先判断根节点是否为空
int count = targetSum - root->val;//根节点也算在路径中,所以count也要减去其值
return traversal(root, count);
}
};

迭代法

TBD

注意点

1、在主函数中,必须确认根节点是否为空,否则必错

2、在主函数中,不要忘记把根节点的值加入递减计数

二刷问题

递归函数的停止条件顺序不能互换
class Solution {
public:
//使用递归,遍历出所有路径,将路径之和与target比较
//确定递归函数的参数和返回值
bool traversal(TreeNode* node, int count){
//确定终止条件
//到叶子节点并找到路径
if(node->left == NULL && node->right == NULL && count == 0) return true;
//到叶子节点但没找到路径
if(node->left == NULL && node->right == NULL) return false;
//错误:
//if(node->left == NULL && node->right == NULL) return false;
//if(node->left == NULL && node->right == NULL && count == 0) return true;

这是因为当根节点没有左右子树时,需要特别处理。

在第一个if语句中,如果count等于0,则说明从根节点到叶子节点的路径上所有节点之和正好等于目标值targetSum,因此返回true。在第二个if语句中,如果没有匹配到第一个if语句的条件,则说明到达了叶子节点但是总和不等于目标值targetSum,此时应该返回false,表示没有符合条件的路径。

如果将两个if语句的顺序互换,会导致遍历到叶子节点时无法正确判断是否存在一条符合条件的路径。当叶子节点没有左右子树时,先进入第二个if语句中的条件判断,返回false,而不检查路径和是否等于目标值,因此会误判。

也就是说,到了叶子节点必须先判断一路上的路径值,如果满足条件就要返回true,如果上述代码顺序颠倒,则会把一些正确的满足条件的结果漏掉,直接判断为false,从而导致错误

记得回溯

到当前叶子节点的路径不满足条件时要往回走,记得要再把count加回来,也就是进行回溯操作

路径总和II

力扣题目链接(opens new window)

给定一个二叉树和一个目标和,找到所有从根节点到叶子节点路径总和等于给定目标和的路径。

说明: 叶子节点是指没有子节点的节点。

示例: 给定如下二叉树,以及目标和 sum = 22,

思路

因为题目要求的是返回所有路径节点和满足target的路径

所以,我们需要遍历整棵树

按照前面的讨论,这种情况下,递归函数不需要返回值(因为不需要满足条件就返回)

这里只介绍递归法,迭代法太麻烦了,写了也记不住

本题中,用于存放路径的数组path和存放结果的二维数组res需要定义在递归函数外

1、确定递归函数的参数和返回值

路径总和的基本思路一致,输入是根节点和一个计数变量(通过递减来寻找满足条件的路径)

返回值是不需要的

2、确定终止条件

终止条件也与路径总和一致,共两种:

  • 遇到叶子节点,且路径满足条件
  • 遇到叶子节点但不满足条件

3、确定单层逻辑

在这一步中,我们除了对计数变量count进行递减和回溯,还需要将当前节点的值加入路径数组path中,并在回溯时将元素pop掉

代码

class Solution {
public:
vector<vector<int>> res;//存放最值输出结果
vector<int> path;//存放当前满足条件的路径
//确定递归函数的参数和返回值
//输入参数为根节点和计数变量
//本题中,需要找出所有路径节点之和满足条件的路径,因此需要遍历整棵树,故没有返回值
void traversal(TreeNode* node, int count){
//确定终止条件
//遇到叶子节点,且路径满足条件
if(node->left == NULL && node->right == NULL && count == 0){
//保存此时path中记录的路径
res.push_back(path);
return;
}
//遇到叶子节点但不满足条件
if(node->left == NULL && node->right == NULL) return; //确定单层逻辑
if(node->left){
//保存当前节点值到path
path.push_back(node->left->val);
//计数递减
count -= node->left->val;
traversal(node->left, count);//调用递归
//回溯,包括计数变量和路径数组
count += node->left->val;
path.pop_back();
} if(node->right){
//保存当前节点值到path
path.push_back(node->right->val);
//计数递减
count -= node->right->val;
traversal(node->right, count);//调用递归
//回溯,包括计数变量和路径数组
count += node->right->val;
path.pop_back();
}
} vector<vector<int>> pathSum(TreeNode* root, int targetSum) {
//确认根节点不为空
if(root == NULL) return res;
int count = targetSum - root->val;//将根节点加入递减计数
path.push_back(root->val);//将根节点值加入数组
traversal(root, count);
return res;
}
};

注意点

1、用于存放路径的数组path和存放结果的二维数组res需要定义在递归函数外

2、在主函数中,必须确认根节点是否为空

3、在主函数中,不要忘记把根节点的值加入递减计数,以及路径数组

求根节点到叶节点数字之和

https://leetcode.cn/problems/sum-root-to-leaf-numbers/

给你一个二叉树的根节点 root ,树中每个节点都存放有一个 0 到 9 之间的数字。

每条从根节点到叶节点的路径都代表一个数字:

例如,从根节点到叶节点的路径 1 -> 2 -> 3 表示数字 123 。

计算从根节点到叶节点生成的 所有数字之和 。

叶节点 是指没有子节点的节点。

示例 1:

输入:root = [1,2,3]

输出:25

解释:

从根到叶子节点路径 1->2 代表数字 12

从根到叶子节点路径 1->3 代表数字 13

因此,数字总和 = 12 + 13 = 25

示例 2:

输入:root = [4,9,0,5,1]

输出:1026

解释:

从根到叶子节点路径 4->9->5 代表数字 495

从根到叶子节点路径 4->9->1 代表数字 491

从根到叶子节点路径 4->0 代表数字 40

因此,数字总和 = 495 + 491 + 40 = 1026

提示:

树中节点的数目在范围 [1, 1000] 内

0 <= Node.val <= 9

树的深度不超过 10

思路

路径总和II类似,因为要遍历完二叉树的所有路径,因此递归函数无需返回值,只需去操控一个vector即可

在本题中,以何种方式遍历二叉树是无所谓的

下面来三部曲

确定递归函数+终止条件

前面说了,因为要遍历整个树,所以递归函数不需要返回值。显然,其输入是当前节点

class Solution {
private:
void traversal(TreeNode* cur){
if(cur->left == nullptr && cur->right == nullptr){//遍历到叶子节点
res += vec2Int(path);
return;
} }
public:
int sumNumbers(TreeNode* root) {
}
};

当前节点为叶子节点时,将一路上收集到的各个节点的值(path数组)转换为整数

(注意叶子节点的判断依据:即左右子节点为空)

此时,需要一个辅助函数vec2Int,可以注意一下该函数的实现,这是一种整数转换的方法

class Solution {
private:
int res = 0;
vector<int> path;
int vec2Int(vector<int>& path){//用于处理数组path,将其转换为整数
int sum = 0;
for(int i = 0; i < path.size(); ++i){
sum = sum*10 + path[i];
}
return sum;
}
void traversal(TreeNode* cur){
if(cur->left == nullptr && cur->right == nullptr){//遍历到叶子节点
res += vec2Int(path);
return;
} }
public:
int sumNumbers(TreeNode* root) {
}
};
确定单层处理逻辑

这里就跟路径总和II的处理方式一样了,先判断左右子节点是否存在,如果存在就将其值加入路径数组

class Solution {
private:
int res = 0;
vector<int> path;
int vec2Int(vector<int>& path){//用于处理数组path,将其转换为整数
int sum = 0;
for(int i = 0; i < path.size(); ++i){
sum = sum*10 + path[i];
}
return sum;
} void traversal(TreeNode* cur){
if(cur->left == nullptr && cur->right == nullptr){//遍历到叶子节点
res += vec2Int(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) {
}
};

注意记得回溯

代码

class Solution {
private:
int res = 0;
vector<int> path;
int vec2Int(vector<int>& path){//用于处理数组path,将其转换为整数
int sum = 0;
for(int i = 0; i < path.size(); ++i){
sum = sum*10 + path[i];
}
return sum;
} void traversal(TreeNode* cur){
if(cur->left == nullptr && cur->right == nullptr){//遍历到叶子节点
res += vec2Int(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) {
if(root == nullptr) return 0;
path.push_back(root->val);//不要忘了先把当前节点压入数组
traversal(root);
return res;
}
};

【LeetCode二叉树#09】路径总和I+II,以及求根节点到叶节点数字之和(回溯回溯,还是™的回溯)的更多相关文章

  1. 【LeetCode】113. 路径总和 II

    题目 给定一个二叉树和一个目标和,找到所有从根节点到叶子节点路径总和等于给定目标和的路径. 说明: 叶子节点是指没有子节点的节点. 示例: 给定如下二叉树,以及目标和 sum = 22, 5 / \ ...

  2. 【LeetCode】437. 路径总和 III

    437. 路径总和 III 给定一个二叉树,它的每个结点都存放着一个整数值. 找出路径和等于给定数值的路径总数. 路径不需要从根节点开始,也不需要在叶子节点结束,但是路径方向必须是向下的(只能从父节点 ...

  3. 【LeetCode】112. 路径总和 Path Sum 解题报告(Java & Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 回溯 BFS 栈 日期 题目地址:https ...

  4. [LeetCode 112 113] - 路径和I & II (Path Sum I & II)

    问题 给出一棵二叉树及一个和值,检查该树是否存在一条根到叶子的路径,该路径经过的所有节点值的和等于给出的和值. 例如, 给出以下二叉树及和值22: 5         / \       4  8  ...

  5. LeetCode 5129. 下降路径最小和 II Minimum Falling Path Sum II

    地址 https://leetcode-cn.com/contest/biweekly-contest-15/problems/minimum-falling-path-sum-ii/ 题目描述给你一 ...

  6. LeetCode【112. 路径总和】

    思路就是从根节点开始向下选节点,依次与sum比较大小,若小,则向下选左右节点其中一个,若大,则接下来判断是否是叶子节点,若是,则返回false 若不是,则上一步选另一节点,再将上述重新执行. 对于叶子 ...

  7. lintcode:二叉树的路径和

    题目 给定一个二叉树,找出所有路径中各节点相加总和等于给定 目标值 的路径. 一个有效的路径,指的是从根节点到叶节点的路径. 解题 下面有个小bug 最后比较的时候是叶子节点为空,左右都有叶子结点,所 ...

  8. LintCode-376.二叉树的路径和

    二叉树的路径和 给定一个二叉树,找出所有路径中各节点相加总和等于给定 目标值 的路径. 一个有效的路径,指的是从根节点到叶节点的路径. 样例 给定一个二叉树,和 目标值 = 5: 返回: [      ...

  9. Java实现求二叉树的路径和

    题: 解: 这道题考的是如何找出一个二叉树里所有的序列. 我的思路是先从根节点开始遍历,找出所有的子节点,因为每个子节点只有一个父节点,再根据每个子节点向上遍历找出所有的序列,再判断序列的总和. 这样 ...

  10. [Leetcode] Path Sum路径和

    Given a binary tree and a sum, determine if the tree has a root-to-leaf path such that adding up all ...

随机推荐

  1. Bitmap、RoaringBitmap原理分析

    作者:京东科技 曹留界 在人群本地化实践中我们介绍了人群ID中所有的pin的偏移量可以通过Bitmap存储,而Bitmap所占用的空间大小只与偏移量的最大值有关系.假如现在要向Bitmap内存入两个p ...

  2. web开发的模式的介绍与身份认证

    web开发的模式的介绍 1.服务端渲染 2.前端端分离开发的web模式 服务端渲染优点与缺点 优点: 1.前端耗时少.因为服务器端负责动态生成HTML内容,浏览器只需要直接渲染页面即可.尤其是移动端更 ...

  3. axios发送请求时携带token

    请求头携带token async getUserlist(){ // 需要授权的Api,必须在青丘头中使用Authorization 字段提供token令牌 const AUTH_TOKEN=loca ...

  4. ts函数讲解与不确定参数是否使用以及限制返回类型

    1.声明式函数 函数有两个参数 参数的数据类型是 string number 返回值值 number 哈 function person(name: string, age: number): num ...

  5. 玩一玩golang汇编

    作者:张富春(ahfuzhang),转载时请注明作者和引用链接,谢谢! cnblogs博客 zhihu Github 公众号:一本正经的瞎扯 因为只是玩一玩,所以走的路线是:用C写代码,把C编译成AT ...

  6. C/C++ 实现切片免杀的思路

    今天突然想到了一个好玩的免杀思路,原理就是想办法切断磁盘特征与内存特征,关于沙盒免杀我寻思着,这样可以将不同的的DLL映射到内存,在内存中他们的特征也是被切断的,在注入器上做判断如果是沙盒则不加载,不 ...

  7. C/C++ ShellCode 常用加密方式

    异或加密ShellCode: #include <stdio.h> #include <Windows.h> unsigned char buf[] = "\xba\ ...

  8. OpenGL的深度缓冲

      如果我们想要在三维空间里画两个正方形:一个红色的,一个绿色的,而且从人眼的观察角度看,绿色正方形在红色正方形的后面,最后看上去应该是这样的: 要点在于,从观察者的角度看,绿色正方形在红色正方形的后 ...

  9. electron 安装不同的版本的方法

    1.官网:http://www.electronjs.org/ 2.思考,既然是npm 安装,那么肯定也在 npm中央仓库有,那么去中央仓库看下: npm i -D electron@11.0.4

  10. 开源.NetCore通用工具库Xmtool使用连载 - 发送短信篇

    [Github源码] <上一篇> 介绍了Xmtool工具库中的发送邮件类库,今天我们继续为大家介绍其中的发送短信类库. 发送短信就像发送邮件一样,在软件系统中使用非常普遍,甚至比发送邮件还 ...