首页 > 其他分享 >回溯——7.子集II

回溯——7.子集II

时间:2024-09-06 21:52:18浏览次数:20  
标签:used nums 元素 II 子集 result 回溯 path

力扣题目链接

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

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

示例:

  • 输入: [1,2,2]
  • 输出: [ [2], [1], [1,2,2], [2,2], [1,2], [] ]

解题思路总结:

  1. 排序:首先对数组进行排序,便于之后的重复元素跳过处理。
  2. 回溯法:通过递归遍历所有可能的子集,并在每次递归中将当前路径加入结果集。
  3. 去重:利用排序后的数组,结合 used 数组,通过条件 nums[i] == nums[i-1]not used[i - 1],来跳过同一层中重复的元素,从而避免生成重复子集。

完整代码如下:

class Solution:
    def subsetsWithDup(self, nums):
        result = []
        path = []
        used = [False] * len(nums)
        nums.sort()  # 去重需要排序
        self.backtracking(nums, 0, used, path, result)
        return result

    def backtracking(self, nums, startIndex, used, path, result):
        result.append(path[:])  # 收集子集
        for i in range(startIndex, len(nums)):
            # used[i - 1] == True,说明同一树枝 nums[i - 1] 使用过
            # used[i - 1] == False,说明同一树层 nums[i - 1] 使用过
            # 而我们要对同一树层使用过的元素进行跳过
            if i > 0 and nums[i] == nums[i - 1] and not used[i - 1]:
                continue
            path.append(nums[i])
            used[i] = True
            self.backtracking(nums, i + 1, used, path, result)
            used[i] = False
            path.pop()
def subsetsWithDup(self, nums):
    result = []
    path = []
    used = [False] * len(nums)
    nums.sort()  # 去重需要排序
    self.backtracking(nums, 0, used, path, result)
    return result
  • result:用于存储所有不重复的子集。
  • path:用于存储当前正在构建的子集。
  • used:这是一个布尔数组,用来标记数组中的每个元素是否已经在当前路径(path)中使用过。
  • nums.sort():在处理重复元素时,排序是必须的,因为只有在数组有序的情况下,才能通过简单的条件判断去除重复子集。
def backtracking(self, nums, startIndex, used, path, result):
    result.append(path[:])  # 收集子集
  • backtracking 函数是回溯算法的核心。
  • startIndex:控制下一步递归从哪里开始选择元素。
  • 每次递归时,当前的 path(表示当前正在构建的子集)会被复制并加入到 result 中,表示我们收集了一个子集。
for i in range(startIndex, len(nums)):
    # used[i - 1] == True,说明同一树枝 nums[i - 1] 使用过
    # used[i - 1] == False,说明同一树层 nums[i - 1] 使用过
    # 而我们要对同一树层使用过的元素进行跳过
    if i > 0 and nums[i] == nums[i - 1] and not used[i - 1]:
        continue
  • 这里的 for 循环遍历数组的每一个元素,从 startIndex 开始。
  • if i > 0 and nums[i] == nums[i - 1] and not used[i - 1]:这个条件判断用于跳过重复的元素,以避免生成重复的子集。具体解释如下:
    • i > 0:确保访问 nums[i-1] 时不会越界。
    • nums[i] == nums[i - 1]:如果当前元素和前一个元素相同,则可能会生成重复子集。
    • not used[i - 1]:前一个元素如果在同一层中没有被使用过(即没有在当前路径中被选择),则说明我们在当前层次中遇到了重复元素,此时应该跳过,以防止生成重复子集。
    path.append(nums[i])
    used[i] = True
    self.backtracking(nums, i + 1, used, path, result)
    used[i] = False
    path.pop()
  • path.append(nums[i]):将当前元素加入到当前路径中。
  • used[i] = True:标记当前元素已经在当前路径中使用。
  • self.backtracking(nums, i + 1, used, path, result):递归调用,开始从下一个索引 i + 1 继续构造子集。
  • used[i] = False:回溯后,将当前元素标记为未使用,以便在其他路径中使用。
  • path.pop():回溯的关键步骤,撤销之前的选择,恢复状态,以便继续构造其他子集。

标签:used,nums,元素,II,子集,result,回溯,path
From: https://blog.csdn.net/plutomty/article/details/141970060

相关文章

  • DFS算法专题(一)——二叉树中的深搜【回溯与剪枝的初步注入】
    目录1、DFS算法简介2、算法实战应用【leetcode】2.1计算布尔二叉树的值2.1.1算法原理 2.1.2算法代码2.2求根节点到叶节点数字之和  2.2.1算法原理​2.2.2算法代码2.3二叉树剪枝2.3.1算法原理2.3.2算法代码2.4验证二叉搜索树 2.4.1算法原理 2.4.2......
  • 软设每日打卡——霍夫曼编码将频繁出现的字符釆用短编码,出现频率较低的字符采用长编码
    【题目】霍夫曼编码将频繁出现的字符釆用短编码,出现频率较低的字符采用长编码。具体        的操作过程为:i)以每个字符的出现频率作为关键字构建最小优先级队列;ii)取出关键        字最小的两个结点生成子树,根节点的关键字为孩子节点关键字之和,并将根节点......
  • Leetcode面试经典150题-210.课程表II
    这个题是图的问题,因为图的拓扑排序在实际应用中有非常多的用途图,所以最近考的越来越多解法都在代码里,不懂就留言或者私信看这个题之前一定要好好看看207题我写的题解,也许207看懂了的话,210只是一个coding问题了Leetcode面试经典150题-207.课程表-CSDN博客一定要看!一定要看!......
  • leetcode刷题day8|字符串部分(344.反转字符串、541. 反转字符串II)
    前言:字符串部分还是比较简单的344.反转字符串链接:https://leetcode.cn/problems/reverse-string/description/思路:这个比较简单,因为字符串也是数组类型的,用两个指针进行交换即可。代码如下:classSolution{publicvoidreverseString(char[]s){inti=0......
  • Java毕设项目II基于Java的英语知识应用网站
    目录一、前言二、技术介绍三、系统实现四、论文参考五、核心代码六、源码获取全栈码农以及毕业设计实战开发,CSDN平台Java领域新星创作者,专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末一、前言在全球化日益加深的今天,英语作为国际交流......
  • 代码训练营 Day23| 39. 组合总和 |40.组合总和II |131.分割回文串
    39.组合总和1.组合没有数量要求2.元素可无限重复选取classSolution(object):defbacktracking(self,cadinates,target,sum_,startindex,path,result):#recursionstopconditionifsum_>target:#wecan'tfindanyanswerset......
  • 删除有序数组中的重复项 II
    给你一个有序数组nums,请你原地删除重复出现的元素,使得出现次数超过两次的元素只出现两次,返回删除后数组的新长度。不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(1)额外空间的条件下完成。示例1:输入:nums=[1,1,1,2,2,3]输出:5,nums=[1,1,2,2,3]解释:......
  • 213. 打家劫舍 II(leetcode)
    https://leetcode.cn/problems/house-robber-ii/description/灵神题解:https://leetcode.cn/problems/house-robber-ii/solutions/2445622/jian-ji-xie-fa-zhi-jie-diao-yong-198-ti-qhvri/classSolution{publicintrob(int[]nums){//f[i]表示前i个房屋选......
  • 最大上升子序列 II
    序列:可以不连续,但与原数列当中出现的先后顺序要相同;上升子序列:需要满足单调性-单调递增算法1(贪心+二分)O(nlogn)时间复杂度二分查找一个数的最小的最大值O(logn);一共有n个数进行二分O(nlogn);贪心分析样例:731218561.首先分析长度为1的上升子序列—......