Leetcode刷题详解——子集

1. 题目链接:78. 子集

2. 题目描述:

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

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

示例 1:

复制代码
输入:nums = [1,2,3]
输出:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]

示例 2:

复制代码
输入:nums = [0]
输出:[[],[0]]

提示:

  • 1 <= nums.length <= 10
  • -10 <= nums[i] <= 10
  • nums 中的所有元素 互不相同

3. 解法1:

3.1 算法思路:

为了获取nums数组的所有子集,我们需要对数组的每个元素进行选择或不选择的操作,即nums数组一定存在2^(数组长度)个子集。对于查找子集,具可以定义一个数组,来记录当时的状态,并对其进行递归。

3.2 递归流程:

  1. 递归结束条件:如果当前需要处理的元素下标越界,则记录当前状态并直接返回
  2. 在递归过程中,对于每个元素,我们有两种选择:
    1. 不选择当前元素,直接递归到下一个元素
    2. 选择当前元素,将其添加到数组末尾后递归到下一个元素,然后在递归结束时撤回添加操作
  3. 所有符合条件的状态都将被记录下来,返回即可

3.3 C++算法代码:

c++ 复制代码
class Solution {
    vector<vector<int>> ret; // 存储所有子集的结果
    vector<int> path; // 当前路径(即当前子集)
public:
    vector<vector<int>> subsets(vector<int>& nums) {
        dfs(nums,0); // 从第一个元素开始进行深度优先搜索
        return ret; // 返回所有子集的结果
    }
    void dfs(vector<int>&nums,int pos)
    {
        if(pos==nums.size()) // 如果已经遍历完所有元素
        {
            ret.push_back(path); // 将当前路径(即当前子集)添加到结果中
            return; // 结束当前递归
        }
        // 选
        path.push_back(nums[pos]); // 将当前元素添加到当前路径中
        dfs(nums,pos+1); // 继续向下一层递归,处理下一个元素
        path.pop_back(); // 恢复现场,即移除当前路径中的最后一个元素
        // 不选
        dfs(nums,pos+1); // 继续向下一层递归,处理下一个元素,但不将当前元素添加到当前路径中
    }
};

4. 解法2:

4.1 算法思路:

对于每个元素有两种选择:

  1. 不进行任何操作;

  2. 将其添加至当前状态的集合。

    在递归时我们需要保证递归结束时当前的状态与进行递归操作前的状态不变,而当我们在选择进行步骤2进行递归时,当前状态会发生变化,因此我们需要在递归结束时撤回添加操作,即进行回溯

4.2 C++算法代码:

c++ 复制代码
class Solution {
    vector<vector<int>> ret; // 存储所有子集的结果
    vector<int> path; // 当前路径(即当前子集)
public:
    vector<vector<int>> subsets(vector<int>& nums) {
        dfs(nums,0); // 从第一个元素开始进行深度优先搜索
        return ret; // 返回所有子集的结果
    }
    void dfs(vector<int>&nums,int pos)
    {
        ret.push_back(path); // 将当前路径添加到结果中
        for(int i=pos;i<nums.size();i++) // 遍历从pos开始的剩余元素
        {
            path.push_back(nums[i]); // 将当前元素添加到当前路径中
            dfs(nums,i+1); // 继续向下一层递归,处理下一个元素
            path.pop_back(); // 恢复现场,即移除当前路径中的最后一个元素
        }
    }
};
相关推荐
郝学胜-神的一滴14 小时前
循环队列深度剖析:从算法原理到C++实现全解析
开发语言·数据结构·c++·算法·leetcode
Via_Neo14 小时前
接雨水问题 + 输入优化
java·开发语言·算法
plus4s14 小时前
3月13日(进阶5)
算法
x_xbx14 小时前
LeetCode:27. 移除元素
数据结构·算法·leetcode
云泽80814 小时前
C++ map 底层探秘:从结构设计到 operator [] 实现的全解析
数据结构·c++·算法
小O的算法实验室15 小时前
2026年EAAI SCI1区TOP,基于LLM驱动的多群粒子群算法动态通信策略生成方法,深度解析+性能实测
算法·论文复现·智能算法·智能算法改进
午彦琳15 小时前
leetcode hot 100_49,128
算法·leetcode·职场和发展
郝学胜-神的一滴15 小时前
深度解析:Python元类手撸ORM框架,解锁底层编程魔法
数据结构·数据库·python·算法·职场和发展
big_rabbit050215 小时前
[算法][力扣219]存在重复元素2
数据结构·算法·leetcode
闻缺陷则喜何志丹15 小时前
【构造 前缀和】P8902 [USACO22DEC] Range Reconstruction S|普及+
c++·算法·前缀和·洛谷·构造