https://oj.leetcode.com/problems/path-sum-ii/

Given a binary tree and a sum, find all root-to-leaf paths where each path's sum equals the given sum.

For example:
Given the below binary tree and sum = 22,

              5
/ \
4 8
/ / \
11 13 4
/ \ / \
7 2 5 1

return

[
[5,4,11,2],
[5,8,4,5]
]

解题思路:

这题和上题 Path Sum 很类似,在他的基础上,要求输出所有和为sum的path。上题讲过,这种题目一般用DFS来做,那么在递归DFS的时候,同时维护一个总的结果的List<List<Integer>> resultList,和一个当前可能的子结果List<Integer> currentList。

这道题本该一次性就写出来,无奈犯了一个非常低级的语言层面的错误。符合条件的时候,直接将currentList加入到resultList中了。这样resultList中等于是存了N个一模一样的list对象了。应该每次拷贝currenList生成一个新的对象放入resultList,就不会有问题了。代码如下:

/**
* Definition for binary tree
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode(int x) { val = x; }
* }
*/
public class Solution {
public List<List<Integer>> pathSum(TreeNode root, int sum) {
List<List<Integer>> resultList = new ArrayList<List<Integer>>();
List<Integer> currentList = new ArrayList<Integer>();
DFS(root, sum, 0, resultList, currentList);
return resultList;
} public void DFS(TreeNode root, int sum, int sumCurrent, List<List<Integer>> resultList, List<Integer> currentList){
if(root == null){
return;
} currentList.add(root.val);
sumCurrent = sumCurrent + root.val;
if(root.left == null && root.right == null){
if(sumCurrent == sum){
//之前一直错,犯了低级错误!!!resultList.add(currentList);
resultList.add(new LinkedList(currentList));
}
}
DFS(root.left, sum, sumCurrent, resultList, currentList);
DFS(root.right, sum, sumCurrent, resultList, currentList);
currentList.remove(currentList.size() - 1);
//下面一行可要可不要
// sumCurrent = sumCurrent - root.val;
}
}

可是至少有一个问题,为什么同样是子状态,前面currentList就一定要回溯,currentSum就不要回溯?

上面的算法中,我们看到一个DFS+回溯算法的基本模样,前面提到的另外两道题DFS题目,Letter Combinations of a Phone Number 和 Word Break II 也都是这个模型。可以说,这种算法无论是在面试中还是实际编程中都是相当重要的,理解了它会有种豁然开朗的感觉。我们可以稍微总结一下。

首先,判断递归返回的条件,写在最开始。然后,处理当前节点。处理完当前节点后,DFS下一节点,最后回溯。或者循环处理当前节点内的所有可能的状态(比如上面电话号码簿里一个数字所有代表的字母,这时一般用for循环)。

回溯是这个算法的关键,下面我借用一个网友的伪代码,讲讲自己对上面的问题,也就是回溯,为什么要回溯的理解。引用的网址在下面有。

void dfs(int 当前状态)
{
if(当前状态为边界状态)
{
记录或输出
return;
}
for(i=0;i<n;i++) //横向遍历解答树所有子节点
{
//扩展出一个子状态。
修改了全局变量
if(子状态满足约束条件)
{
dfs(子状态)
}
恢复全局变量//回溯部分
}
}

从上面的伪代码里我们可以看到,对当前节点的一个状态的操作,往往会包含对全局变量的修改,类似于本题中的currentList。想想递归的过程,实际上是一个树形的占空间,每个子状态都有自己的栈,存放当前的临时变量,这也就是为什么有时候递归非常耗费空间的原因。

这时我们再来看上面提出的问题,为什么同样是子状态,前面currentList就一定要回溯,currentSum就不要回溯?这里我只说Java,因为自己对C或者C++不了解。

在Java里,所有方法(或函数)的参数,只有pass by value,这和C++可以传引用是非常不同的。所以所有的primitive type,比如int, char等,传的都是值。在本题中,DFS方法的参数,sum和sumCurrent传的都是他们的数值,而不是本身,他们都是每个递归状态都会维护在自己占空间内的,所以后面的修改不会影响前面栈内的值。这就是currentSum不需要回溯的原因。用上面的伪代码来说,它不是全局变量。最后一行sumCurrent = sumCurrent - root.val;是多余的,因为它减也是减去当前栈内的值,和其他栈并不相关。

反过来,currenList是一个对象。严格来说,Java中,currentList中存的是一个内存地址,这片内存内才住着真正的对象。所以DFS方法传参的时候,pass by value,就将这个地址传了进去。在整个递归的过程中,所有对currenList的修改,都是修改的这一个内存地址,也就是这一个对象。各自自状态的栈空间里,保存的副本也仅仅是这个内存地址的值。各个递归状态对currenList的修改,都会影响其他状态。这也就是为什么currentList需要回溯的原因。dfs修改后,需要立刻恢复它到修改前的状态。

有其他网友总结的更好,我来转载下。

http://blog.csdn.net/fightforyourdream/article/details/12866861

http://blog.chinaunix.net/uid-26602509-id-3178938.html

http://blog.sina.com.cn/s/blog_779fba530100wqz9.html

http://blog.csdn.net/cdfr2321388/article/details/5725863

从这个问题上,我们还能比较好的理解Java里pass by value的实质,为什么primitive就是pass by value,而object type却感觉是pass by reference,实际上也是pass by value。

那么思路来了,我们能不能将这个currenList在每次递归的栈空间里都维护一个副本,这样就不需要回溯了嘛。下面的代码便是这样的一个实现。

/**
* Definition for binary tree
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode(int x) { val = x; }
* }
*/
public class Solution {
public List<List<Integer>> pathSum(TreeNode root, int sum) {
List<List<Integer>> resultList = new ArrayList<List<Integer>>();
List<Integer> currentList = new ArrayList<Integer>();
DFS(root, sum, 0, resultList, currentList);
return resultList;
} public void DFS(TreeNode root, int sum, int sumCurrent, List<List<Integer>> resultList, List<Integer> currentList){
if(root == null){
return;
} //回溯的关键,每次申明一个新的内存放当前状态的结果currentList,后面就不要回溯了
List<Integer> soFarList = new LinkedList(currentList);
soFarList.add(root.val);
sumCurrent = sumCurrent + root.val;
if(root.left == null && root.right == null){
if(sumCurrent == sum){
//之前一直错,犯了低级错误!!!resultList.add(currentList);
resultList.add(new LinkedList(soFarList));
}
}
DFS(root.left, sum, sumCurrent, resultList, soFarList);
DFS(root.right, sum, sumCurrent, resultList, soFarList);
//下面一步就不必要了,这个对理解回溯很重要
// currentList.remove(currentList.size() - 1);
//下面一行可要可不要
// sumCurrent = sumCurrent - root.val;
}
}

应该说,回溯放在递归方法的最后面,理解起来是有点难度的,这样一解释应该能清楚很多。

这类题目在面试中经常出现,以及DFS和BFS的区别,啥时候用啥,各自优缺点之类,需要好好理解。

最新文章

  1. Hark的数据结构与算法练习之堆排序
  2. Apache Spark源码走读之9 -- Spark源码编译
  3. 236. Lowest Common Ancestor of a Binary Tree
  4. 在 App 扩展和主 App 间共享数据
  5. android 拍照或者图库选择 压缩后 图片 上传
  6. Html jquery实现根据 IOS和Android访问跳转
  7. Angular - - Angular数据类型判断
  8. Fis3迁移至Webpack实战
  9. java.lang.Collections
  10. idea用到的快捷键
  11. Visual Studio 2015 将json转换为实体类
  12. PHP 生成验证码(+图片没有显示的解决办法)
  13. shell =~ 引发的思考
  14. ubuntu查看文件和文件夹大小
  15. 比酒量|2012年蓝桥杯B组题解析第三题-fishers
  16. 团队作业——Alpha冲刺 3/12
  17. 内存溢出(Memory Overflow)和内存泄露(Memory Leak)的区别
  18. iOS开发中的压缩以及解压
  19. [UE4]机器人自动寻路
  20. 解除单个文件的与svn服务器的关联

热门文章

  1. layui 时间前后节点验证
  2. python特性小记(一)
  3. QS之force(1)
  4. hibernate中的懒加载和急加载
  5. 使用OpenCV画折线图
  6. (转)Arcgis for Js之鼠标经过显示对象名的实现
  7. java mongodb 使用MongoCollection,BasicDBObject 条件查询
  8. BZOJ 1864: [Zjoi2006]三色二叉树 树形DP + 读入
  9. [luogu1090 SCOI2003] 字符串折叠(区间DP+hash)
  10. ecshop3 调用指定分类下推荐/热卖/新品商品,可指定调用数量