leetcode:90. 子集 II

题目来源

题目描述

在这里插入图片描述

题目解析

如果没有重复元素
在这里插入图片描述
如果重复元素
在这里插入图片描述
在这里插入图片描述

重复的原因是:刚刚选择了,然后撤销了这个选择,之后又选择了和刚刚相同的元素

class Solution {
    std::vector<std::vector<int>> ans;
    std::vector<int> path;
    void dfs(vector<int>& nums, int i){
        if(i == nums.size()){
            ans.push_back(path);
            return;
        }

        path.push_back(nums[i]);
        dfs(nums, i + 1);
        path.pop_back();

        while (i + 1 < nums.size() && nums[i] == nums[i + 1]){
            ++i;
        }
        dfs(nums, i + 1);
    }
public:
    vector<vector<int>> subsetsWithDup(vector<int>& nums) {
        std::sort(nums.begin(), nums.end());
        ans.emplace_back();
        dfs(nums, 0);
        return ans;
    }
};

解题思路

一般情况下,看到题目要求「所有可能的结果」,而不是「结果的个数」,我们就知道需要暴力搜索所有的可行解了,可以用「回溯法」。

「回溯法」实际上一个类似枚举的搜索尝试过程,主要是在搜索尝试过程中寻找问题的解,当发现已不满足求解条件时,就「回溯」返回,尝试别的路径。

回溯法是一种算法思想,而递归是一种编程方法,回溯法可以用递归来实现。

回溯法的整体思路是:搜索每一条路,每次回溯是对具体的一条路径而言的。对当前搜索路径下的的未探索区域进行搜索,则可能有两种情况:

  • 当前未搜索区域满足结束条件,则保存当前路径并退出当前搜索;
  • 当前未搜索区域需要继续搜索,则遍历当前所有可能的选择:如果该选择符合要求,则把当前选择加入当前的搜索路径中,并继续搜索新的未探索区域。

上面说的未搜索区域是指搜索某条路径时的未搜索区域,并不是全局的未搜索区域。

回溯法搜所有可行解的模板一般是这样的:

res = []
path = []

def backtrack(未探索区域, res, path):
    if path 满足条件:
        res.add(path) # 深度拷贝
        # return  # 如果不用继续搜索需要 return
    for 选择 in 未探索区域当前可能的选择:
        if 当前选择符合要求:
            path.add(当前选择)
            backtrack(新的未探索区域, res, path)
            path.pop()

backtrack 的含义是:未探索区域中到达结束条件的所有可能路径,path 变量是保存的是一条路径,res 变量保存的是所有搜索到的路径。所以当「未探索区域满足结束条件」时,需要把 path 放到结果 res 中。

path.pop() 是啥意思呢?它是编程实现上的一个要求,即我们从始至终只用了一个变量 path,所以当对 path 增加一个选择并 backtrack 之后,需要清除当前的选择,防止影响其他路径的搜索。

对于求不含重复元素的数组的子集,思路如下:

  • 未搜索区域:剩余的未搜索的数组nums[indexx : N - 1]
  • 每个path是否满足题目的条件:任何一个path的子集,都满足条件,都要放到res中
  • 当前path满足条件时,是否继续搜索:使得,找到nums[0:index-1]中的子集之后,nums[index]添加到老的path中形成新的子集
  • 未探索区域当前可能的选择:每次选择可以选取 s 的 1 个字符,即nums[index]
  • 当前选择符合要求:任何 nums[index] 都是符合要求的,直接放到 path 中;
  • 新的未探索区域:nums 在 index 之后的剩余字符串, nums[index + 1 : N - 1] 。

上面分析了那么多,让我们看看代码怎么写。由于每个path都是一个子集,都满足条件,所以就省略了一个 if 判断,并且由于当前 path 满足条件时,应该继续搜索,所以也省略了 if 中的 return 语句。

代码如下:

class Solution {
    std::vector<std::vector<int>> ans;
    std::vector<int> path;
    void dfs(vector<int>& nums, int idx){
        ans.push_back(path);
        for (int i = idx; i < nums.size(); ++i) {
            path.push_back(nums[i]);
            dfs(nums, idx + 1);
            path.pop_back();
        }
    }
public:
    vector<vector<int>> subsetsWithDup(vector<int>& nums) {
        dfs(nums, 0);
        return ans;
    }
};

对于本题,含有重复元素,比如说求 nums = [1,2,2] 的子集,那么对于子集 [1,2] 是选择了第一个 2,那么就不能再选第二个 2 来构成 [1,2] 了。所以,此时的改动点,就是先排序,每个元素 nums[i] 添加到 path 之前,判断一下 nums[i] 是否等于 nums[i - 1] ,如果相等就不添加到 path 中。

class Solution {
    std::vector<std::vector<int>> ans;
    std::vector<int> path;
    void dfs(vector<int>& nums, int idx){
        if(idx > nums.size()){
            return;
        }

        ans.push_back(path);
        for (int i = idx; i < nums.size(); ++i) {
            if (i != idx && nums[i] == nums[i - 1]) continue;
            path.push_back(nums[i]);
            dfs(nums, idx + 1);
            path.pop_back();
        }
    }
public:
    vector<vector<int>> subsetsWithDup(vector<int>& nums) {
        sort(nums.begin(), nums.end());
        dfs(nums, 0);
        return ans;
    }
};
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值