39. 组合总和
做了46题之后,大概明白了回溯搜索是怎么回事,紧接着我就回过头来做了这道题(怎么感觉有点回溯那味儿了?哈哈哈可能这就是缘分吧),结果一把过!我分析了一下,这道题的基本框架跟46-全排列差不多,都要搜索所有的情况,但是有一些不一样的地方:
1. 终止的条件不一样:46题中要把给定数组中的数字都用一遍,所以终止条件是一个答案(path)中的元素个数等于给定数组的元素个数(length);而在这里,我们要找到一组目标元素,使得加起来等于目标值,所以最后的终止条件应该跟目标值target有关;
2. 递归部分不一样:这里的元素可以重复使用,所以for循环的部分肯定不一样。
但可以肯定的是,肯定有回溯的过程。
总的思路: 回溯搜索 + 无重复添加(不会反向添加)
1. 终止条件的确定
由于是求和,且每次选取一个数,所以我就想到了,每次将target的值减去选取的nums[i]的值,直到最后这个target小于等于0了,表明再也找不到数了,因此递归终止。
在递归终止的时候,也比较容易想到两种不同的情况。不像46题中那样,条件终止时,找到的集合肯定是满足条件的;在本题中,找到的可能不满足条件,因为全部的加起来小于了目标值target,因此需要判断target的最后值是否为0,如果是0,那么表示加起来等于原来的目标值,这种情况才应该加入最后的结果中;否则,不应该加入。但是终止条件里面,最后都应该返回了。这部分的代码比较简单,也很容易想到,如下:
if (target <= 0) {
if (target == 0) { // 只有目标值最后为0的时候&#