给定一个无重复元素的数组 candidates
和一个目标数 target
,找出 candidates
中所有可以使数字和为target
的组合。
candidates
中的数字可以无限制重复被选取。
说明:
所有数字(包括
target
)都是正整数。 解集不能包含重复的组合。
示例 1:
输入: candidates = [2,3,6,7], target = 7, 所求解集为: [[7],[2,2,3]]
示例 2:
输入: candidates = [2,3,5], target = 8, 所求解集为: [[2,2,2,2],[2,3,3],[3,5]]
答案参考:
思路: 回溯法搜索(排列树)
step 1 : 排序
step 2 : 回溯搜索 , 将每一个可行解保存到res集合中
剪枝方案:
1.经过排序后 如 2, 3, 6 ,7 先选一个数,如果这个数加上已经在可行解中的数没有超过target 如 (sum:4)+2 <= (target:7)说明这个数有可能产生可行解(sum=target) 将其加入到可行解中(2,2,2). 如果这个数加上已经在可行解中的数超过了target 如 (sum:6)+2 <= (target:7)说明这个数已经不可能产生可行解了
2.每次选择的数都是从candidates中选择的[2,3,6,7](遍历方式,从小到大) 如果2+sum>target , 那么还有必要尝试选择3,6,7吗?显然没有必要了. 3.根据题目要求.产生的可行解也是递增的 如[2,2,3] 可以 [3,2,2] 不可以 [3,2,3] 不可以 也就是说,如果当前选择的数为
3,那么下一层就不能选择2 如果当前选择的数为6,那么下一层就不能选择2,3 (通过遍历时对起始下标的控制能很好的解决这个问题)
/**
* @param {number[]} candidates
* @param {number} target
* @return {number[][]}
*/
var combinationSum = function(candidates, target) {
var item=[],path=[];
no_repetition(candidates,target,0,item,path);
function no_repetition(candidates,target,it,item,path){
if(target<0)
return;
if(target==0){
path=path.slice()
item.push(path);
return
}
for(var i=it;i<candidates.length;i++){
path.push(candidates[i]);
no_repetition(candidates,target-candidates[i],i,item,path)
path.pop()
}
}
return item
};