首先要说明二叉树的问题就是用递归来做,基本没有其他方法,因为这数据结构基本只能用递归遍历,不要把事情想复杂了。

#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); // 如果当前节点能够满足条件,有一条路径了,但是还是要继续搜索左右子树,因为值不全为非负,这里返回的路径都具有同一个根节点。
}
};

这里有两个递归函数,函数pathSumreturn sumUp(root, 0, sum) + pathSum(root->right, sum) + pathSum(root->left, sum);是以该节点root为起点的路径条数、以root->right为起点或者以其子孙节点为起点的路径条数、以root->left为起点或者以其子孙节点为起点的路径条数。

[LeetCode] #112 #113 #437 Path Sum Series的更多相关文章

  1. LeetCode 112. 路径总和(Path Sum) 10

    112. 路径总和 112. Path Sum 题目描述 给定一个二叉树和一个目标和,判断该树中是否存在根节点到叶子节点的路径,这条路径上所有节点值相加等于目标和. 说明: 叶子节点是指没有子节点的节 ...

  2. 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 ...

  3. 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 ...

  4. 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 ...

  5. 【leetcode】437. Path Sum III

    problem 437. Path Sum III 参考 1. Leetcode_437. Path Sum III; 完

  6. 437. Path Sum III

    原题: 437. Path Sum III 解题: 思路1就是:以根节点开始遍历找到适合路径,以根节点的左孩子节点开始遍历,然后以根节点的右孩子节点开始遍历,不断循环,也就是以每个节点为起始遍历点 代 ...

  7. Leetcode 931. Minimum falling path sum 最小下降路径和(动态规划)

    Leetcode 931. Minimum falling path sum 最小下降路径和(动态规划) 题目描述 已知一个正方形二维数组A,我们想找到一条最小下降路径的和 所谓下降路径是指,从一行到 ...

  8. [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 ...

  9. [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 ...

随机推荐

  1. Dubbo学习(五) Dubbo 从下载到编译成功

    DUBBO是一个分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案,是阿里巴巴SOA服务化治理方案的核心框架,每天为2,000+个服务提供3,000,000,000+次访问量支持,并被广 ...

  2. JAVA LOG4J使用方法

    首先,需要在项目中导入log4j使用的JAR包,导入结果如下图: 菜单:Build Path->Configure Build Path->Add Extern Jars 导入JAR包后, ...

  3. SpringBoot整合Kotlin构建Web服务

    今天我们尝试Spring Boot整合Kotlin,并决定建立一个非常简单的Spring Boot微服务,使用Kotlin作为编程语言进行编码构建. 创建一个简单的Spring Boot应用程序.我会 ...

  4. dp的一些计划

    抱歉这是鸽子贴. 树形dp [x][[POI2014]HOT-Hotels](https://www.luogu.org/problemnew/show/P3565) [x][[HAOI2015]树上 ...

  5. 解题:USACO13NOV No Change

    题面 在朴素中透着一点新意的状压DP 一个很暴力的思路是枚举位置,状态和硬币,每次二分出向前最多能买到哪里,复杂度爆炸($O(2^knklog$ $n)$) 考虑优化,不妨先预处理一下$goal[i] ...

  6. Webpack + React 开发 02 JSX 语法

    HTML 语言直接写在 JavaScript 语言之中,不加任何引号,这就是 JSX 的语法,它允许 HTML 与 JavaScript 的混写: render(<h1>Hello Wor ...

  7. 【DP】【P4539】 [SCOI2006]zh_tree

    Description 张老师根据自己工作的需要,设计了一种特殊的二叉搜索树. 他把这种二叉树起名为zh_tree,对于具有n个结点的zh_tree,其中序遍历恰好为(1,2,3,-,n),其中数字1 ...

  8. 数据库之Oracle的介绍与使用20180620

    /*******************************************************************************************/ 一.orac ...

  9. linux基础命令之sed

    什么是sed? sed是一个流编辑器,英文全名为stream editor,流的意思就是能处理文件或者通过管    道传输来的数据.主要针对行,也就是行编辑.默认是不编辑文件的,处理时,把当前    ...

  10. GO_08:GO语言基础之interface

    接口interface 1. 接口是一个或多个方法签名的集合 2. 只要某个类型拥有该接口的所有方法签名,即算实现该接口,无需显示声明实现了哪个接口,这称为 Structural typing 3. ...