【递归、搜索与回溯】穷举,暴搜,深搜,回溯,剪枝:全排列与子集

文章目录

1. 全排列(LC46)

全排列

题目描述

解题思路

定义全局变量int[][] ret记录结果,int[] path记录路径。利用决策树来辅助:

定义全局变量boolean[] check判断数组中的元素是否被使用过

  • 把数组中所有元素枚举一遍,调用check判断,如果使用过则直接排除(剪枝);没有被使用过则添加到path中。
  • 回溯:函数调用完,要把path的最后一个元素删去,并且在check中标记为false,"恢复现场"
  • 递归出口:遇到"叶子节点"直接返回。

代码实现

java 复制代码
class Solution {
    List<List<Integer>> ret = new ArrayList<>();
    List<Integer> path = new ArrayList<>();
    boolean[] check;

    public List<List<Integer>> permute(int[] nums) {
        check = new boolean[nums.length];
        dfs(nums);
        return ret;
    }
    void dfs(int[] nums){
        if(nums.length == path.size()){
            ret.add(new ArrayList<>(path));
            return;
        }

        for(int i = 0;i<nums.length;i++){
            if(check[i])
                continue;
            path.add(nums[i]);
            check[i] = true;
            dfs(nums);
            //回溯 -》 恢复现场
            path.removeLast();
            check[i] = false;
        }
    }
}

注意:
ret.add(new ArrayList<>(path))不可以写成ret.add(path),因为第二种方式添加的是引用。当回溯path清空,ret中的内容也会清空。必须新建一个ArrayList对象

2. 子集(LC78)

子集

题目描述

解题思路

与上一题相同,定义全局变量path记录路径,全局变量ret记录结果。

  • 解法一:

    1. 借助决策树来判断:每一个节点可以选择数组中的元素是否存在。
    2. 函数头:void dfs(int nums[],int i) i表示当前要考虑数组中第i个元素是否要存在。
      • 不选:直接继续调用dfs(nums,i+1);
      • 选择当前元素:path先添加当前元素,再调用dfs(nums,i+1)
    3. 递归出口:当i==nums.length-1,说明已经遍历完所有元素,就可以退出了。
  • 解法二:

    1. 第一层考虑一个元素,第二层考虑两个元素,以此类推。
    2. 往后扩展时只能从当前元素后面选择,否则会重复。决策树的节点就是结果。
    3. 函数头:void dfs(int[] nums,int i)i 表示当前函数从第i个元素开始枚举

代码实现

  • 解法一:
java 复制代码
class Solution {
    List<List<Integer>> ret = new ArrayList<>();
    List<Integer> path = new ArrayList<>();
    public List<List<Integer>> subsets(int[] nums) {
        dfs(nums,0);
        return ret;
    }
    void dfs(int[] nums,int i){
        if(i == nums.length){
            ret.add(new ArrayList(path));
            return;
        }
        //不选当前元素
        dfs(nums,i+1);

        //选择当前元素
        path.add(nums[i]);
        dfs(nums,i+1);
        //恢复现场
        path.removeLast();
    }
}
  • 解法二:
java 复制代码
class Solution {
    List<List<Integer>> ret = new ArrayList<>();
    List<Integer> path = new ArrayList<>();
    public List<List<Integer>> subsets(int[] nums) {
        dfs(nums,0);
        return ret;
    }
    void dfs(int[] nums,int i){
        //每一个节点都是一个结果
        ret.add(new ArrayList<>(path));
        for(int j = i;j<nums.length;j++){
            path.add(nums[j]);
            dfs(nums,j+1);
            //恢复现场
            path.removeLast();
        }
    }
}
相关推荐
枫叶林FYL几秒前
安装Claude Code泄密
人工智能·机器学习
计算机安禾4 分钟前
【数据结构与算法】第20篇:二叉树的链式存储与四种遍历(前序、中序、后序、层序)
c语言·开发语言·数据结构·c++·学习·算法·visual studio
菜菜的顾清寒17 分钟前
力扣HOT100(16)除了自身以外数组的乘积
算法·leetcode·职场和发展
sali-tec17 分钟前
C# 基于OpenCv的视觉工作流-章45-圆卡尺
图像处理·人工智能·opencv·算法·计算机视觉
AI数据皮皮侠18 分钟前
支持向量机
人工智能·机器学习·支持向量机
会编程的土豆29 分钟前
【数据结构与算法】二叉树遍历 集合
数据结构·算法
老虎062729 分钟前
LeetCode热题100 刷题笔记(第六天)双指针 「 盛最多水的容器」
笔记·算法·leetcode
ZPC821030 分钟前
fanuc 机器人 pr 寄存器数据如何绑定 GO 绑定DO
人工智能·算法·机器人
LUVK_40 分钟前
第六章 图
数据结构·考研·算法·深度优先·图论·408
逆境不可逃1 小时前
【用AI学Agent】Agent入门前置:大模型基础(开发向)
人工智能·深度学习·机器学习