Idiot-maker

  :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

https://leetcode.com/problems/combination-sum/

Given a set of candidate numbers (C) and a target number (T), find all unique combinations in C where the candidate numbers sums to T.

The same repeated number may be chosen from C unlimited number of times.

Note:

  • All numbers (including target) will be positive integers.
  • Elements in a combination (a1, a2, … , ak) must be in non-descending order. (ie, a1 ≤ a2 ≤ … ≤ ak).
  • The solution set must not contain duplicate combinations.

 

For example, given candidate set 2,3,6,7 and target 7
A solution set is: 
[7] 
[2, 2, 3] 

解题思路:

这仍然是一道permutation的题目,考虑用DFS。

掌握了前面N道DFS的题目,这里的思路应该非常清晰了。遇到的问题有两个,第一,结果的数字要升序。第二,结果不能有重复的组合。

第一个问题比较好解决,处理开始就将数组排序即可。

第二个问题要考虑下,方法一:将结果算出来,塞进List<List<Integer>>的时候看是不是已经存在。方法二:避免算出重复的结果。我这里用了第二种方法,其实和前面visited数组的标记很像。

跑一下未处理重复的程序可以发现,重复的答案都是,[2, 2, 3]和[3, 2, 2]这种。所以递归的时候,只选取当前元素往后的元素就可以了。往前的因为已经选过,可以不选。这样就避免了重复的答案。

public class Solution {
    public List<List<Integer>> combinationSum(int[] candidates, int target) {
        List<List<Integer>> resultList = new LinkedList<List<Integer>>();
        List<Integer> currentList = new LinkedList<Integer>();
        //原题没说数组已经排序了,必须先排序
        Arrays.sort(candidates);
        dfs(resultList, currentList, 0, 0, candidates, target);
        return resultList;
    }
    
    public void dfs(List<List<Integer>> resultList, List<Integer> currentList, int currentSum, int step, int[] candidates, int target){
        if(currentSum == target){
            resultList.add(new LinkedList(currentList));
            return;
        }
        if(currentSum > target){
            return;
        }
        //避免结果重复的方法:i从当前元素开始往后开始,所以要把当前元素作为参数传进方法
        for(int i = step; i < candidates.length; i++){
            currentList.add(candidates[i]);
            currentSum += candidates[i];
            dfs(resultList, currentList, currentSum, i, candidates, target);
            currentList.remove(currentList.size() - 1);
            currentSum -= candidates[i];
        }
    }
}

仍然要注意的是,必须是resultList.add(new LinkedList(currentList)),而不能直接resultList.add(currentList)。

最后要注意特别重要的一点,题目给出的是,Given a set of candidate numbers (C) and a target number (T),也就是数组中是没有重复元素的,否则例如[1,1,1,1,1],1。即便从当前元素往后搜索,仍然会有重复答案。

posted on 2015-03-15 20:12  NickyYe  阅读(172)  评论(0编辑  收藏  举报