路径总和 III:不需要从根节点查找,返回满足条件的个数
#leetcode
#算法/二叉树
#算法/前缀和
目录
1. 总结
- 路径前缀和
- 遍历一遍的思路
var pathSum = function (root, targetSum) {
// key 是前缀和,value 是前缀和为 key 的个数
const mapping = new Map();
// 初始化,前缀和为 0 的路径有一条,即"空路径"
mapping.set(0, 1);
let sum = 0; // 从根节点到当前节点的路径和
let res = 0;
const traverse = function (root) {
if (root == null) return;
sum += root.val;
res += mapping.get(sum - targetSum) || 0; // 更新 res
mapping.set(sum, (mapping.get(sum) || 0) + 1);
traverse(root.left);
traverse(root.right);
mapping.set(sum, mapping.get(sum) - 1);
sum -= root.val;
};
traverse(root);
return res;
};
2. 题目及理解
3. 解题思路
- 二叉树的前序后序遍历
- 前缀和技巧
4. 代码实现
/**
* @param {TreeNode} root
* @param {number} targetSum
* @return {number}
*/
var pathSum = function(root, targetSum) {
// key 是前缀和,value 是前缀和为 pathSum 的个数
const preSumCount = new Map();
// 初始化,前缀和为 0 的路径有一条
// 为什么要初始化呢?
// 这个设置可以被理解为代表一个"空路径",其和为 0
preSumCount.set(0, 1);
// pathSum 记录当前路径和,即从根节点到当前节点的路径和
let pathSum = 0;
// res 记录满足条件的路径条数
let res = 0;
const traverse = function(root) {
// 递归终止条件
// base case
if (root == null) {
return;
}
/*************************************************
* ::::::::::::::::::::: 前序遍历位置 ::::::::::::::::::
************************************************/
// 前序遍历位置, 计算路径和
pathSum += root.val;
// 先看一下路径和为 pathSum - targetSum 的路径有多少条
// :::: pathSum - targetSum 代表的是从根节点到当前节点的路径和为 targetSum
// 即满足条件的路径,所以更新 res 的值
res += preSumCount.get(pathSum - targetSum) || 0;
// 记录从二叉树的根节点开始,路径和为 pathSum 的路径条数
preSumCount.set(pathSum, (preSumCount.get(pathSum) || 0) + 1);
traverse(root.left);
traverse(root.right);
/*************************************************
* ::::::::::::::: 后序遍历位置 ::::::::::::::
************************************************/
preSumCount.set(pathSum, preSumCount.get(pathSum) - 1);
pathSum -= root.val;
}
traverse(root);
return res;
};
5. 复杂度分析
- 时间复杂度:O(N),其中 N 是树中的节点数。
- 空间复杂度:O(N),主要由递归调用栈和
preSumCount
Map 贡献。
6. 错误记录
初始化空路径的处理:preSumCount.set(0, 1);