leetcode-78-子集

36 篇文章 1 订阅
28 篇文章 0 订阅
78. 子集

给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。

解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。

示例 1:

输入:nums = [1,2,3]
输出:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]

示例 2:

输入:nums = [0]
输出:[[],[0]]

方法一:迭代法实现子集枚举

思路与算法

因为在这数组中没有重复的元素,所有我们可以利用二进制的0,1表示元素是否出现,从而进行组合。

记原序列中元素的总数为 n。原序列中的每个数字 ai 的状态可能有两种,即「在子集中」和「不在子集中」。我们用 1 表示「在子集中」,0 表示不在子集中,那么每一个子集可以对应一个长度为 n 的 0/1 序列,第 i 位表示 ai 是否在子集中。例如,n=3 ,a={5,2,9} 时:

可以发现 0/1 序列对应的二进制数正好从 000 到 2^n - 1。我们可以枚举 mask∈[0,2n−1],mask 的二进制表示是一个 0/1 序列,我们可以按照这个 0/1 序列在原集合当中取数。当我们枚举完所有 2^n 个 mask,我们也就能构造出所有的子集。

代码实现
class Solution {
    List<Integer> t = new ArrayList<Integer>();
    List<List<Integer>> ans = new ArrayList<List<Integer>>();

    public List<List<Integer>> subsets(int[] nums) {
        int n = nums.length;
        //1 << n  1左移n位确定循环的次数
        for (int mask = 0; mask < (1 << n); ++mask) {
            //每次刷新t
            t.clear();
            for (int i = 0; i < n; ++i) {
                //比较mask的二进制的每一位是否为1,是1就加入相应位置的数
                if ((mask & (1 << i)) != 0) {
                    t.add(nums[i]);
                }
            }
            //将t加入集合
            ans.add(new ArrayList<Integer>(t));
        }
        return ans;
    }
}

方法二:递归法实现子集枚举

思路与算法

我们也可以用递归来实现子集枚举。

假设我们需要找到一个长度为 n 的序列 a 的所有子序列,代码框架是这样的:

List<Integer> t = new ArrayList<Integer>();
public void dfs(int cur, int[] nums) {
        if (cur == nums.length) {
            // 记录答案
        	// ...
        }
    	// 考虑选择当前位置
        t.add(nums[cur]);
        dfs(cur + 1, nums);
    	// 考虑不选择当前位置
        t.remove(t.size() - 1);
        dfs(cur + 1, nums);
    }

对于 cur 位置,我们需要考虑 a[cur] 取或者不取,如果取,我们需要把 a[cur]放入一个临时的答案数组中(即上面代码中的 t,再执行 dfs(cur+1,n),执行结束后需要对 t 进行回溯;如果不取,则直接执行 dfs(cur+1,n)。当 cur 增加到 n 的时候,记录答案并终止递归。

class Solution {
    List<Integer> t = new ArrayList<Integer>();
    List<List<Integer>> ans = new ArrayList<List<Integer>>();

    public List<List<Integer>> subsets(int[] nums) {
        //开始递归,从位置0开始
        dfs(0, nums);
        return ans;
    }

    public void dfs(int cur, int[] nums) {
        if (cur == nums.length) {
            // 记录答案
            ans.add(new ArrayList<Integer>(t));
            return;
        }
        // 考虑选择当前位置
        t.add(nums[cur]);
        dfs(cur + 1, nums);
        // 不考虑选择当前位置
        t.remove(t.size() - 1);
        dfs(cur + 1, nums);
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值