[LeetCode] #112 #113 #437 Path Sum Series
首先要说明二叉树的问题就是用递归来做,基本没有其他方法,因为这数据结构基本只能用递归遍历,不要把事情想复杂了。
#112 Path Sum
原题链接:https://leetcode.com/problems/path-sum/ 。
判断从树的根节点到叶子节点的路径中,是否有一条所有节点上的值之和和特定的数字,即sum。
从根节点到叶子节点,线路的起点的是固定的,只需要不断递归下去,判断在叶子节点处是否满足根节点加到该节点的值之和为sum。
这个限制条件把这个问题简化了很多很多。
/**
* 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 {
public:
bool hasPathSum(TreeNode* root, int sum) {
if (!root) return false;
if (root->val == sum && root->left == NULL && root->right == NULL) return true;
return hasPathSum(root->left, sum - root->val) || hasPathSum(root->right, sum - root->val);
}
};
#113 Path Sum II
与前题要做的事情一致,只是现在要求输出路径。如果路径有多条,要求输出多条路径。
要输出路径,在递归的过程中肯定是要有一个传址的变量vector<int>将路径记录下来。有多条路径,需要把多条路径vector<vector<int> >记录下来。
/**
* 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 {
public:
vector<vector<int>> pathSum(TreeNode* root, int sum) {
vector<vector<int> > paths;
vector<int> path;
pathSumHelper(root, sum, path, paths);
return paths;
}
private:
void pathSumHelper(TreeNode* root, int sum, vector<int>& path, vector<vector<int>>& paths) {
if (root == NULL) return;
path.push_back(root->val);
if (root->val == sum && root->left == NULL && root->right == NULL) {
paths.push_back(path);
}
pathSumHelper(root->left, sum - root->val, path, paths);
pathSumHelper(root->right, sum - root->val, path, paths);
path.pop_back();
}
};
这里有一点搞不懂,为什么最后要进行path.pop_back(),如果我把代码写成这样子:
void pathSumHelper(TreeNode* root, int sum, vector<int>& path, vector<vector<int>>& paths) {
if (root == NULL) return;
path.push_back(root->val);
if (root->left == NULL && root->right == NULL) {
if (root->val == sum) {
paths.push_back(path);
}
path.pop_back();
}
pathSumHelper(root->left, sum - root->val, path, paths);
pathSumHelper(root->right, sum - root->val, path, paths);
}
然后,发现如果这么写我是错的。对于下面的这一棵二叉树,得到的结果是[[5,4,11,2],[5,4,11,8,4,5]],正确结果是[[5,4,11,2],[5,8,4,5]]。第二条路径中多了4, 11,问题出在只是把根节点去除掉,没有去除中间节点,使得中间节点及其子孙节点中的中间节点出现在了经过其兄弟节点的路径。
如,左侧的4节点会留在右子树的路径中是因为,调用根节点的pathSumHelper时,会调用两次pathSumHelper(root->left/* node 4 */, sum - root->val, path, paths);、pathSumHelper(root->right/* node 8 */, sum - root->val, path, paths);,在第一次调用完成之后,path变量中的4节点未被删除,所以存留在了路径[5,4,11,8,4,5]中(就是第一个4)。
#437 Path Sum III
继续,判断是否存在路径上的值之和为特定的数字sum。变化在于起点不固定、终点也不固定,值存在负数(其实无所谓,因为前面的代码没有判断数值超过sum就不再递归)。最后的输出是路径的条数。
/**
* 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 {
public:
int pathSum(TreeNode* root, int sum) {
if (!root) return 0;
return sumUp(root, 0, sum) + pathSum(root->right, sum) + pathSum(root->left, sum);
}
private:
int sumUp(TreeNode* root, int pre, int sum) {
if (!root) return 0;
int current = pre + root->val;
return (current == sum) + sumUp(root->right, current, sum) + sumUp(root->left, current, sum); // 如果当前节点能够满足条件,有一条路径了,但是还是要继续搜索左右子树,因为值不全为非负,这里返回的路径都具有同一个根节点。
}
};
这里有两个递归函数,函数pathSum的return sumUp(root, 0, sum) + pathSum(root->right, sum) + pathSum(root->left, sum);是以该节点root为起点的路径条数、以root->right为起点或者以其子孙节点为起点的路径条数、以root->left为起点或者以其子孙节点为起点的路径条数。
[LeetCode] #112 #113 #437 Path Sum Series的更多相关文章
- LeetCode 112. 路径总和(Path Sum) 10
112. 路径总和 112. Path Sum 题目描述 给定一个二叉树和一个目标和,判断该树中是否存在根节点到叶子节点的路径,这条路径上所有节点值相加等于目标和. 说明: 叶子节点是指没有子节点的节 ...
- LeetCode(113) Path Sum II
题目 Given a binary tree and a sum, find all root-to-leaf paths where each path's sum equals the given ...
- leetcode 112. Path Sum 、 113. Path Sum II 、437. Path Sum III
112. Path Sum 自己的一个错误写法: class Solution { public: bool hasPathSum(TreeNode* root, int sum) { if(root ...
- 47. leetcode 437. Path Sum III
437. Path Sum III You are given a binary tree in which each node contains an integer value. Find the ...
- 【leetcode】437. Path Sum III
problem 437. Path Sum III 参考 1. Leetcode_437. Path Sum III; 完
- 437. Path Sum III
原题: 437. Path Sum III 解题: 思路1就是:以根节点开始遍历找到适合路径,以根节点的左孩子节点开始遍历,然后以根节点的右孩子节点开始遍历,不断循环,也就是以每个节点为起始遍历点 代 ...
- Leetcode 931. Minimum falling path sum 最小下降路径和(动态规划)
Leetcode 931. Minimum falling path sum 最小下降路径和(动态规划) 题目描述 已知一个正方形二维数组A,我们想找到一条最小下降路径的和 所谓下降路径是指,从一行到 ...
- [LeetCode] 437. Path Sum III 路径和 III
You are given a binary tree in which each node contains an integer value. Find the number of paths t ...
- [LeetCode] 437. Path Sum III_ Easy tag: DFS
You are given a binary tree in which each node contains an integer value. Find the number of paths t ...
随机推荐
- 关于sizeof
sizeof是求占用的内存空间的大小,并不是指数组长度.(strlen 的长度只适合char*类型) 例如. int a[10]={0}; 数组a的长度为sizeof(a)/sizeof(a[0])— ...
- maven依赖jar包时版本冲突的解决
https://blog.csdn.net/sinat_39789638/article/details/78005945 共有四种解决方式: 1.第一声明优先原则: 在pom.xml配置文件中,如果 ...
- 一道面试题:StringBuffer a=new StringBuffer ("A"); StringBuffer b=new StringBuffer
前几天又看到这个面试题,再次看看 public class Jtest{ public static void main(String[] args) { StringBuffer a=new Str ...
- URAL 1969. Hong Kong Tram
有一个trick就是没想到,枚举第二段时间后,要检测该火车能否继续跑一圈来判断,不能先检测前半圈能不能跑加进去后在检测后半段: // **** 部分不能放在那个位置: 最近代码导致的错误总是找不出,贴 ...
- pragma指令详解(转载)
#pragma comment( comment-type [,"commentstring"] ) 该宏放置一个注释到对象文件或者可执行文件.comment-type是一个预定义 ...
- 【BZOJ1045】糖果传递(贪心)
[BZOJ1045]糖果传递(贪心) 题面 BZOJ 洛谷 题解 秉承者娱乐精神,我们必须写一个费用流,并且相信信仰跑不过去. 于是写了一个\(zkw\)费用流如下:(您可以无视此份代码) #incl ...
- 一步步创建第一个Docker App —— 3. 创建一个集群Swarm
原文:https://docs.docker.com/engine/getstarted-voting-app/create-swarm/ 初始化集群 Swarm 1. 使用 ssh 命令登录 man ...
- 面向对象高级编程(1)-使用__slots__
使用__slots__ 正常情况下,当我们定义了一个class,创建了一个class的实例后,我们可以给该实例绑定任何属性和方法,这就是动态语言的灵活性.先定义class: >>> ...
- Linux下vim 快捷键
vim按d表示剪切 按dd剪切一行 vim命令:命令模式 /关键字 n继续向下查找vim的多行注释: 1.按ctrl + v进入 visual block模式 2.按上下选中要注释的行 3.按大写字母 ...
- R语言 线性回归
0 引言 初学者,对于一些运行结果不是很清楚,所以看了一些课本和资料,这里做一个记录而已. 1 线性回归模型的结果分析 结果的解释: “call”:指出线性回归的公式 “Residuals”:之处从实 ...