题目:
给定一个可能有重复数字的整数数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。
candidates 中的每个数字在每个组合中只能使用一次,解集不能包含重复的组合
示例 :
输入: candidates = [10,1,2,7,6,1,5], target = 8, 输出: [ [1,1,6], [1,2,5], [1,7], [2,6] ]
思路:
还是老思路,回溯剪枝,唯一需要注意的是怎么不重复使用candidates数组中的数字。
先给数组排序,同级递归中,不能碰到两个一样的数,因为后面的数组合肯定是前一个数的子集。
复杂度:
时间:快排O(2。
空间:没有额外空间,复杂度O(1)。
代码:
public List<List<Integer>> combinationSum2(int[] candidates, int target) {
Arrays.sort(candidates);
List<List<Integer>> res = new ArrayList<>();
int L = candidates.length;
dfs(L,candidates,res,0,target,new ArrayList<>());
return res;
}
private void dfs(int L,int[] candidates,List<List<Integer>> res,int idx,
int target,List<Integer> cur){
if(target == 0){
res.add(new ArrayList<>(cur));
return;
}
for (int i = idx; i < L; i++) {
if (candidates[i] > target) { // 剪枝
break;
}
if (i > idx && candidates[i] == candidates[i - 1]) { // 剪枝、避免重复
// 因为前面那个同样的数已经经历过dfs,再拿同样的数dfs就会得到重复的答案
continue;
}
cur.add(candidates[i]);
dfs(L,candidates,res, i + 1, target - candidates[i], cur);
cur.remove(cur.size() - 1);
}
}