LeetCode OJ 78. Subsets

Given a set of distinct integers, nums, return all possible subsets.

Note:

  • Elements in a subset must be in non-descending order.
  • The solution set must not contain duplicate subsets.

 For example,

If nums = [1,2,3], a solution is:

[
  [3],
  [1],
  [2],
  [1,2,3],
  [1,3],
  [2,3],
  [1,2],
  []
]

【题目分析】

这个题目是求一个集合的子集,要求子集中的元素不能降序排列,而且不含有重复的元素。


【思路】

求一个集合的所有子集我们可以考虑子集中包含集合的若干个元素,从0个到n(集合的大小)。如果含有0个元素,就是空集,含有一个元素的话就是把集合中每个元素取一次,含有两个元素的话就是找到集合中所有两个元素的组合......以此类推。

问题是这个过程该如何描述呢?假设我们要从一个增序排序的集合中选取所有两个元素的组合,从集合中开始选取一个之前没有选过的元素a,然后再从这个元素后面选择一个元素b。当我们开始找下一个组合时,需要回溯到当前组合,删除元组中的第二个元素b,然后继续向后寻找。同样当我们找到所有第一个元素为a的所有二元子集时,我们需要删除a,然后在a后面找到下一个二元组的开始元素a2,然后继续上面的过程。因此这个过程是一个递归回溯的过程。

回溯法

回溯法按深度优先策略搜索问题的解空间树。首先从根节点出发搜索解空间树,当算法搜索至解空间树的某一节点时,先利用剪枝函数判断该节点是否可行(即能得到问题的解)。如果不可行,则跳过对该节点为根的子树的搜索,逐层向其祖先节点回溯;否则,进入该子树,继续按深度优先策略搜索。

回溯法的基本行为是搜索,搜索过程使用剪枝函数来为了避免无效的搜索。剪枝函数包括两类:1. 使用约束函数,剪去不满足约束条件的路径;2.使用限界函数,剪去不能得到最优解的路径。

问题的关键在于如何定义问题的解空间,转化成树(即解空间树)。解空间树分为两种:子集树和排列树。两种在算法结构和思路上大体相同。


【java代码】

 1 public class Solution {
 2     public List<List<Integer>> subsets(int[] nums) {
 3         List<List<Integer>> list = new ArrayList<>();
 4         List<Integer> clist = new ArrayList<>();
 5         list.add(clist);
 6         if(nums == null) return list;
 7         Arrays.sort(nums);
 8         
 9         for(int i = 1; i <= nums.length; i++){
10             clist.clear();
11             DFS(nums, 0, i, clist, list);
12         }
13         return list;
14     }
15     
16     public void DFS(int[] nums, int start, int number, List<Integer> clist, List<List<Integer>> list){
17         if(number == clist.size()){
18             list.add(new ArrayList<>(clist));
19             return;
20         }
21         for(int i = start;i < nums.length; i++) {
22             clist.add(nums[i]);
23             DFS(nums, i+1, number, clist, list);
24             clist.remove(clist.size()-1);
25         }
26     }
27 }

 

 1 public class Solution {
 2     public List<List<Integer>> subsets(int[] nums) {
 3         Arrays.sort(nums);
 4         List<List<Integer>> ret = new ArrayList<>();
 5         dfs(nums, 0, new ArrayList<>(), ret);
 6         return ret;
 7         
 8     }
 9 
10     private void dfs(int[] nums, int idx, List<Integer> path, List<List<Integer>> ret) {
11         ret.add(path);
12         for (int i = idx; i < nums.length; i++) {
13             List<Integer> p = new ArrayList<>(path);
14             p.add(nums[i]);
15             dfs(nums, i+1, p, ret);
16         }
17     }
18 }

 

posted @ 2016-06-03 09:26  Black_Knight  阅读(551)  评论(0编辑  收藏  举报