回溯——6.子集

力扣题目链接

给定一组不含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。

说明:解集不能包含重复的子集。

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

解题思路总结

  • 使用回溯法(backtracking)来解决子集问题,其核心思路是对每一个元素做两种选择:要么将其加入到当前的子集中,要么不加入,并不断递归处理剩下的元素。
  • 在每一次递归调用时,都会将当前构造的 path 加入到 result 中,确保所有可能的子集都被枚举出来。
  • 最终返回的 result 包含了所有的子集,包括空集。

完整代码如下:

python 复制代码
class Solution:
    def subsets(self, nums):
        result = []
        path = []
        self.backtracking(nums, 0, path, result)
        return result

    def backtracking(self, nums, startIndex, path, result):
        result.append(path[:])  # 收集子集,要放在终止添加的上面,否则会漏掉自己
        # if startIndex >= len(nums):  # 终止条件可以不加
        #     return
        for i in range(startIndex, len(nums)):
            path.append(nums[i])
            self.backtracking(nums, i + 1, path, result)
            path.pop()
python 复制代码
class Solution:
    def subsets(self, nums):
        result = []
        path = []
        self.backtracking(nums, 0, path, result)
        return result
  • nums:输入的数组,表示需要找出其所有子集的集合。
  • result:用于存储最终结果,即所有可能的子集。
  • path:用来存储当前递归路径上选择的元素,代表一个子集。

在这个函数中,首先定义了 resultpath,分别用来存储最终所有子集的集合和当前正在构建的子集,然后调用 backtracking 函数开始回溯。backtracking 函数会从 nums 的第一个元素(下标为 0)开始递归地探索所有子集。

python 复制代码
    def backtracking(self, nums, startIndex, path, result):
        result.append(path[:])  # 收集子集
        for i in range(startIndex, len(nums)):
            path.append(nums[i])  # 选择元素加入子集
            self.backtracking(nums, i + 1, path, result)  # 递归调用
            path.pop()  # 撤销选择,回溯
2.1 result.append(path[:])
  • 这行代码表示:将当前路径 path 中的元素加入到 result 中。注意这里的 path[:] 是一个浅拷贝,确保后续对 path 的修改不会影响已经加入 result 的内容。
  • 由于任何一个时刻的 path 都可以作为一个合法的子集,因此每当进入 backtracking 函数时,无论 path 中有多少元素,都会将它作为一个子集存入 result
2.2 for 循环
  • for i in range(startIndex, len(nums)):表示从 nums 中当前的起始位置 startIndex 开始,依次选择元素加入到当前子集(即 path)中。
  • 对于每一个元素 nums[i],首先将其加入到 path 中,然后递归调用 backtracking 函数,开始探索从 i + 1 位置开始的下一个元素。
  • 递归完毕后,通过 path.pop() 将最后加入的元素删除,进行回溯,尝试其他的可能组合。
2.3 终止条件
  • 代码中没有显式的终止条件。事实上,回溯函数的退出是通过遍历 nums 数组的长度自动完成的。当 startIndex 达到 len(nums) 时,for 循环不再执行,因此递归自然结束。
  • 可以加一个显式的终止条件,但实际上没有这个必要,因为 for 循环本身已经会停止。
相关推荐
蹉跎x8 分钟前
力扣1358. 包含所有三种字符的子字符串数目
数据结构·算法·leetcode·职场和发展
巫师不要去魔法部乱说1 小时前
PyCharm专项训练4 最小生成树算法
算法·pycharm
IT猿手2 小时前
最新高性能多目标优化算法:多目标麋鹿优化算法(MOEHO)求解GLSMOP1-GLSMOP9及工程应用---盘式制动器设计,提供完整MATLAB代码
开发语言·算法·机器学习·matlab·强化学习
阿七想学习2 小时前
数据结构《排序》
java·数据结构·学习·算法·排序算法
王老师青少年编程2 小时前
gesp(二级)(12)洛谷:B3955:[GESP202403 二级] 小杨的日字矩阵
c++·算法·矩阵·gesp·csp·信奥赛
Kenneth風车2 小时前
【机器学习(九)】分类和回归任务-多层感知机(Multilayer Perceptron,MLP)算法-Sentosa_DSML社区版 (1)111
算法·机器学习·分类
eternal__day2 小时前
数据结构(哈希表(中)纯概念版)
java·数据结构·算法·哈希算法·推荐算法
APP 肖提莫3 小时前
MyBatis-Plus分页拦截器,源码的重构(重构total总数的计算逻辑)
java·前端·算法
OTWOL3 小时前
两道数组有关的OJ练习题
c语言·开发语言·数据结构·c++·算法
qq_433554543 小时前
C++ 面向对象编程:递增重载
开发语言·c++·算法