LeetCode78 子集(带扩展)

leetcode.cn/problems/su...

解法一:回溯

因为集合中的元素不用考虑顺序,我们通过保证元素之间的相对顺序不变来防止出现重复的子集

例如,[1,2,3]2 后面只有 3,如果你添加了前面的 1,那么后面就只能选择23,若选择 了2,后面就只能选择3,否则,如果出现[2,1],就会和之前已经生成的子集 [1,2] 重复了。

得到下面这棵决策树

这棵树的特性是:如果把根节点作为第 0 层,那么第 n 层的所有节点就是大小为 n 的所有子集

题意要求得到所有子集,那只要遍历这棵多叉树,把所有节点的值加入结果即可

go 复制代码
func subsets(nums []int) [][]int {
	path := make([]int, 0)
	var res [][]int
	backtrack(nums, 0, path, &res)
	return res
}

func backtrack(nums []int, offset int, path []int, res *[][]int) {
	if len(path) <= len(nums) { // 每个节点的值都是一个子集
		tmp := make([]int, len(path))
		copy(tmp, path)
		*res = append(*res, tmp)
	}
	for i := offset; i < len(nums); i++ { // 递归退出条件: offset == len(nums)
		path = append(path, nums[i]) // 做选择
		backtrack(nums, i+1, path, res) // 通过offset参数控制树枝的遍历,每次只能往后选择元素,避免产生重复的子集
		path = path[:len(path)-1] // 撤销选择
	}
}

扩展

组合问题和子集问题其实是等价的 ,给定一个数组,大小为 k 的组合就是大小为 k 的子集。

来看这道题:leetcode.cn/problems/co...

回想刚才的决策树,其实这个题就是找出第k层的所有节点,前面的子集代码稍作改动即可

go 复制代码
func combine(n int, k int) [][]int {
    nums := make([]int, 0, n)
    for i:=1; i<=n; i++{
        nums = append(nums, i)
    }
    path := make([]int, 0)
    var res [][]int 
    backtrack(nums, 0, path, k, &res)
    return res
}

func backtrack(nums []int, offset int, path []int, k int, res *[][]int){
    if len(path) == k{ // 收集长度为k的子集
        tmp := make([]int, len(path))
        copy(tmp, path)
        *res = append(*res, tmp)
        return
    }
    for i:= offset; i<len(nums); i++{ // 通过offset避免重复选择元素
        path = append(path, nums[i]) // 做选择
        backtrack(nums, i+1, path, k, res)
        path = path[:len(path)-1] // 撤销选择
    }
}

可以再优化一下空间复杂度,不需要生成一个额外的数组nums

go 复制代码
func combine(n int, k int) [][]int {
    path := make([]int, 0)
    var res [][]int
    backtrack(n, 1, path,k, &res) // 范围是【1,n】
    return res
}

func backtrack(n int, offset int, path []int, k int, res *[][]int) {
    if len(path) == k{
        *res = append(*res, append([]int{}, path...))
        return
    }
    for i:=offset; i<=n; i++{
        path = append(path, i)
        backtrack(n, i+1, path, k, res)
        path = path[:len(path)-1]
    }
}
相关推荐
Dragon Wu33 分钟前
Spring Security Oauth2.1 授权码模式实现前后端分离的方案
java·spring boot·后端·spring cloud·springboot·springcloud
一个有梦有戏的人1 小时前
Python3基础:进阶基础,筑牢编程底层能力
后端·python
爬山算法1 小时前
Hibernate(88)如何在负载测试中使用Hibernate?
java·后端·hibernate
独断万古他化1 小时前
【Spring 原理】Bean 的作用域与生命周期
java·后端·spring
我爱加班、、2 小时前
Websocket能携带token过去后端吗
前端·后端·websocket
愚者游世2 小时前
Delegating Constructor(委托构造函数)各版本异同
开发语言·c++·程序人生·面试·改行学it
一 乐2 小时前
校园二手交易|基于springboot + vue校园二手交易系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端
80530单词突击赢2 小时前
SpringBoot整合SpringMVC全解析
java·spring boot·后端
hdsoft_huge2 小时前
1panel面板中部署SpringBoot和Vue前后端分离系统 【图文教程】
vue.js·spring boot·后端
lekami_兰3 小时前
RabbitMQ 延迟队列实现指南:两种方案手把手教你搞定
后端·rabbitmq·延迟队列