组合总和
leetcode链接
1.解法
这道题和普通的组合问题的区别在于它可以重复取集合中的数值,所以我们再代码实现的时候需要注意这一点。
因为组合问题就是回溯问题,而回溯问题都可以画成树,所以我们先画出图帮助分析:
从图中我们看出一个规律,就是当取得了一个元素后,我们在下一次选择数值的时候,集合就变为了该数值往后的数值集合(包括该数)。举个例子,如果取了2,那么下一次就要在[2,5,3]中选。如果取了5,下一次就要在[5,3]中取。这为我们实现逻辑部分的代码提供了依据。
然后是回溯三部曲:
-
确定函数头和一些辅助变量:
result = [] # 存储最终稿答案path = [] # 存储当前取得的所有数值def backtracking(candidates,target,sum)
因为参数是要在编写代码中确定的,所以我这里先写上这三个参数,后面看完代码大家就懂了。但这里还是简单说一下含义,
candidates代表的是当前取数的集合,
target代表的是目标数值,
sum代表的是已取得的数的总和。
-
确定递归出口:
if sum > target:returnif sum == target:result.append(path)return
(1)当已取得的数值的和,即sum,比目标数值大的时候,就直接退出该层递归;
(2)当已取得的数值的和,即sum,和目标数值一样大的时候,就可以把取得的数值保存一下,然后退出该层递归。
-
确定递归逻辑:
for i in range(len(candidates)):path.append(candidates[i])sum = sum + candidates[i]backtracking(candidates[i:],target,sum)sum = sum - candidates[i]path.pop()
每次递归都要从当前的集合中取数,所以用for循环来完成取数操作,然后每次都要把取得的数值存入path中,sum加上对应的数值,然后递归,注意此时的candidates要变成该元素之后的所有元素组成的集合(包括该元素),然后回溯,将sum减去加上的值,path弹出对应的值。
总体代码为:
def combinationSum(candidates, target):result = []path = []def backtracking(candidates,target,sum):if sum >target:returnif sum == target:path1 = path.copy() result.append(path1)returnfor i in range(len(candidates)):path.append(candidates[i])sum = sum + candidates[i]backtracking(candidates[i:],target,sum)sum = sum - candidates[i]path.pop()backtracking(candidates,target,0)return result
2.优化
我们考虑如果我们将candidates从小到大排序后,那么我们从最小的数值按照顺序开始取,这样如果有一次取得的元素和刚好等于或者大于了target,那么后面的该层循环后面的值都不用再取了,因为一定会大于target。
所以我们可以在逻辑部分加一个判断,如果当sum>target的时候,那么就直接break就好了,不过前提是要先将candidates从小到大排序。
所以修改后的代码为:
def combinationSum(candidates, target):result = []path = []candidates = sorted(candidates)def backtracking(candidates,target,sum):if sum >target:returnif sum == target:path1 = path.copy()result.append(path1)returnfor i in range(len(candidates)):path.append(candidates[i])sum = sum + candidates[i]if sum > target: sum = sum - candidates[i] path.pop()breakbacktracking(candidates[i:],target,sum)sum = sum - candidates[i]path.pop()backtracking(candidates,target,0)return result
3.总结
剪枝不一定是在原有基础上剪枝,有可能还需要先对集合或者其他要素进行一定操作之后才可以剪枝。