DFS算法专题(二)——穷举vs暴搜vs深搜vs回溯vs剪枝【OF决策树】

news2025/1/23 3:52:27

目录

1、决策树

2、算法实战应用【leetcode】

2.1 题一:全排列

2.2.1 算法原理

2.2.2 算法代码

2.2 题二:子集

2.2.1 算法原理【策略一】

2.2.2 算法代码【策略一】

2.2.3  算法原理【策略二,推荐】

2.2.4 算法代码【策略二,推荐】

2.3 题三:找出所有子集的异或总和再求和

2.3.1 算法原理

2.3.2 算法代码【解法一】(不推荐)

 2.3.3 算法代码【解法二】(推荐)

 2.4 题四:全排列 II

2.4.1 算法原理

2.4.2 算法代码


1、决策树

决策树是一种树形结构的监督学习算法,广泛应用于分类任务和回归任务中。它通过递归地将数据集分割成更小的子集,最终形成一个树形模型,用于预测新数据的输出。

决策树是一种树形结构,其中每个内部节点表示一个属性上的测试,每个分支代表一个测试输出,每个叶节点代表一种类别。

决策树是一种十分常用的分类方法。它是一种监督学习,所谓监督学习就是给定一堆样本,每个样本都有一组属性和一个类别,这些类别是事先确定的,那么通过学习得到一个分类器,这个分类器能够对新出现的对象给出正确的分类。这样的机器学习就被称之为监督学习。

面对上难度的DFS算法题,我们首先要做的就是画出决策树。

注意:画出的决策树越详细越好!!!

2、算法实战应用【leetcode】

2.1 题一:全排列

. - 力扣(LeetCode)

2.2.1 算法原理

  1. 画出决策树,记住:决策树越详细越好!!!
  2. 设置全局变量(全局变量的设置因题而异),本题设置二维数组ret为返回值,记录全部的全排列;数组path记录路径;布尔类型的check数组实现剪枝操作
  3. 对于递归函数,我们仍然要站在宏观角度,聚焦于某一层要做的事来设计
  4. DFS算法,不可避免的就是回溯、剪枝与函数出口等细节问题的考虑
  5. 对于本题回溯的设计,在dfs完下一层,回到当前层后,再进行恢复现场的操作(删除下一层在path中的数据),注意:本题需要对path和check均进行回溯操作
  6. 对于本题剪枝的设计,通过布尔数组的判断,判断数组中数据可否放进路径path中
  7. 对于本题递归函数的出口,将path中存放数据的数量和nums数组长度相比较,若相等,则添加到ret中。

2.2.2 算法代码

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

        for(int i = 0; i < nums.length; i++) {
            if(!check[i]) {
                path.add(nums[i]);
                check[i] = true;
                dfs(nums);
                //将下一层中的相关数据删除 -> 回溯 -> 恢复现场 -> 1.path 2.check
                //add新数据时,递归仍然在本层中,add的是本层i位置处的新数据,
                //dfs完后,i仍然为本层add时的数据的下标,
                //此时回溯要将path和check恢复现场
                path.remove(path.size() - 1);
                check[i] = false;
            }
        }
    }
}

2.2 题二:子集

. - 力扣(LeetCode)

2.2.1 算法原理【策略一】

  1. 画决策树
  2. 设置全局变量 : ①:List<List<Integer>> ret;//返回值     ②:List<Integer> path;//记录路径
  3. 设计函数头:void dfs(nums, pos);//pos为下一次进入path中元素的下标
  4. 设计函数体:①:选:path += nums[pos]; dfs(nums,pos+1);②:不选:dfs(nums,pos+1);
  5. 考虑函数出口:pos == nums.length;将path添加进ret中,return;
  6. 处理细节问题:①:回溯 -> 恢复现场:path.remove(pathsize()-1); ②:剪枝:无

2.2.2 算法代码【策略一】

class Solution {
    List<List<Integer>> ret;
    List<Integer> path;
    public List<List<Integer>> subsets(int[] nums) {
        ret = new ArrayList<>();
        path = new ArrayList<>();
        dfs(nums, 0);
        return ret;
    }
    public void dfs(int[] nums, int pos) {
        //函数出口
        if(pos == nums.length) {
            ret.add(new ArrayList<>(path));
            return;
        }
        
        //选
        path.add(nums[pos]);
        dfs(nums, pos + 1);
        //回溯 -> 恢复现场
        path.remove(path.size() - 1);

        //不选(不做任何操作即可)
        dfs(nums, pos + 1);
    }
}

2.2.3  算法原理【策略二,推荐】

思想 

  • 分三个支线,每个节点就是一个子集,所以每刚进入递归函数时,就把path添加进ret中,pos依然为下一个要添加进path中的元素的下标。叶子节点即为函数出口

 步骤:

  1. 画决策树
  2. 设计代码 ①全局变量 ret/path ②函数头 void dfs(nums, pos); ③函数体 1.ret.add(path);2.依次枚举该节点后的元素 ④函数出口  无
  3. 细节问题 ①回溯 -> 恢复现场:path.remove(path,size()-1) ②剪枝 -> 无

2.2.4 算法代码【策略二,推荐】

class Solution {
    List<List<Integer>> ret;
    List<Integer> path;
    public List<List<Integer>> subsets(int[] nums) {
        ret = new ArrayList<>();
        path = new ArrayList<>();
        dfs(nums, 0);
        return ret;
    }
    public void dfs(int[] nums, int pos) {
        ret.add(new ArrayList<>(path));
        //从pos位置开始向后枚举
        for(int i = pos; i < nums.length; i++) {
            //依次枚举,下标为i
            path.add(nums[i]);
            //从当前元素后面的元素中选
            dfs(nums, i + 1);
            //回溯 -> 恢复现场
            path.remove(path.size() - 1);
        }
    }
}

2.3 题三:找出所有子集的异或总和再求和

. - 力扣(LeetCode)

2.3.1 算法原理

本题与上一题解题思想如出一辙,

解法一:直接cv上一题的代码,再将每组子集相异或再求和。(不推荐)

解法二:将path设置为每个子集的异或值(策越二,每个节点都是一个子集),再创建一个全局变量sum求和即可,回溯时直接利用异或消消乐的特性恢复现场即可。

解法二的时间效率更加优秀,建议大家使用解法二解题。

2.3.2 算法代码【解法一】(不推荐)

class Solution {
    List<List<Integer>> total;
    List<Integer> path;
    public int subsetXORSum(int[] nums) {
        total = new ArrayList<>();
        path = new ArrayList<>();
        List<Integer> sum = new ArrayList<>();
        dfs(nums, 0);
        for(int i = 0; i < total.size(); i++) {
            int ret = 0;
            for(int j = 0; j < total.get(i).size(); j++) {
                ret ^= (total.get(i).isEmpty() ? 0 : total.get(i).get(j));
            }
            sum.add(ret);
        }
        int val = 0;
        for(int x : sum) val += x;
        return val;
    }
    public void dfs(int[] nums, int pos) {
        total.add(new ArrayList<>(path));

        for(int i = pos; i < nums.length; i++) {
            path.add(nums[i]);
            dfs(nums, i + 1);
            path.remove(path.size() - 1);
        }
    }
}

 2.3.3 算法代码【解法二】(推荐)

class Solution {
    int path;//每个子集所有元素的异或值
    int sum;
    public int subsetXORSum(int[] nums) {
        path = 0;
        sum = 0;
        dfs(nums, 0);
        return sum;
    }
    public void dfs(int[] nums, int pos) {
        sum += path;

        for(int i = pos; i < nums.length; i++) {
            path ^= nums[i];
            dfs(nums, i + 1);
            //回溯 -> 恢复现场 -> 异或:消消乐
            path ^= nums[i];
        }
    }
}

 2.4 题四:全排列 II

. - 力扣(LeetCode)

2.4.1 算法原理

本题整体框架与上文题一全排列相同,回溯函数出口函数体等设计这里不再赘述。

由于出现重复的元素,故本题算法需要添加更多的剪枝操作。

在数组有序的情况下(有序时相同的元素才会相邻),我们可以这样来设计剪枝:

  1. 剪枝:

①:同一个元素只能出现一次 --> boolean[] check
②:在一个节点的分支中,相同数值的元素只能选择一次

  1. 筛选节点:

符合条件的节点:
check[i]==flase && (i==0 || nums[i-1] != nums[i] || check[i-1]==true)

2.4.2 算法代码

class Solution {
    List<List<Integer>> ret;
    List<Integer> path;
    boolean[] check;//同一个数据,只能出现一次
    public List<List<Integer>> permuteUnique(int[] nums) {
        ret = new ArrayList<>();
        path = new ArrayList<>();
        check = new boolean[nums.length];
        //数据有序的情况下,保证重复的元素是相邻的
        Arrays.sort(nums);
        dfs(nums);
        return ret;
    }
    public void dfs(int[] nums) {
        if(path.size() == nums.length) {
            ret.add(new ArrayList<>(path));
            return;
        }
        for(int i = 0; i < nums.length; i++) {
            //剪枝 -> 筛选出符合条件的分支
            //同一个数据只能出现一次 && 一个节点上,相同数值的元素只能选择一次
            if(!check[i] && (i == 0 || nums[i - 1] != nums[i] || check[i - 1])) {
                path.add(nums[i]);
                check[i] = true;
                dfs(nums);
                //回溯 -> 恢复现场
                path.remove(path.size() - 1);
                check[i] = false;
            }
        }
    }
}

END 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2121055.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【人工智能】Transformers之Pipeline(十六):表格问答(table-question-answering)

​​​​​​​ 目录 一、引言 二、表格问答&#xff08;table-question-answering&#xff09; 2.1 概述 2.2 基于BERT的表格问答模型—TAPAS&#xff08;TAble PArSing&#xff09; 2.3 应用场景​​​​​​​ 2.4 pipeline参数 2.4.1 pipeline对象实例化参数 2.4…

mysql 限制用户登录次数超过3次就 锁定账户在一段时间内不运行操作

这里是引用 主要实现步骤&#xff1a; 1.目测安装的mysql版本得是5.7.40往上&#xff0c;因为我的版本是5.7.14发现里面没有控制等下限制这个插件&#xff0c;插件具体的查看是在你安装目录下的lib/pugin下面 比如我的&#xff1a;C:\zz\ProgramFiles\MySQL\MySQL Server 5.7\l…

【VSCode扩展】通义灵码运行提示“此应用无法在你的电脑上运行”

&#x1f935; 作者&#xff1a;coderYYY &#x1f9d1; 个人简介&#xff1a;前端程序媛&#xff0c;目前主攻web前端&#xff0c;后端辅助&#xff0c;其他技术知识也会偶尔分享&#x1f340;欢迎和我一起交流&#xff01;&#x1f680;&#xff08;评论和私信一般会回&#…

AI模型“减肥”风潮:量化究竟带来了什么?

量化对大模型的影响是什么 ©作者|YXFFF 来源|神州问学 引言 大模型在NLP和CV领域的广泛应用中展现了强大的能力&#xff0c;但随着模型规模的扩大&#xff0c;对计算和存储资源的需求也急剧增加&#xff0c;特别是在资源受限的设备上面临挑战。量化技术通过将模型参数和…

【Transformer】Post-Norm和Pre-Norm

文章目录 Transformer中使用的Post-Norm大模型常用的Pre-NormPre-Norm一定比Post-Norm好吗&#xff1f;二者区别总结参考资料 Pre-Norm和Post-Norm的区别&#xff0c;是面试官非常喜欢问的问题。下面我们按照时间线&#xff0c;尽可能直白地讲清楚二者的区别。 直观来讲&#x…

【MySQL】MySQL表的增删改查(进阶篇)——之约束条件

前言&#xff1a; &#x1f31f;&#x1f31f;本期讲解关于MySQL表增删查改进阶篇&#xff0c;希望能帮到屏幕前的你。 &#x1f308;上期博客在这里&#xff1a;http://t.csdnimg.cn/cF0Mf &#x1f308;感兴趣的小伙伴看一看小编主页&#xff1a;GGBondlctrl-CSDN博客 目录 …

什么是监督学习(Supervised Learning)

一、监督学习概述 监督学习&#xff08;Supervised Learning&#xff09;是一种极具威力的机器学习方法&#xff0c;能够训练算法以识别数据中的模式&#xff0c;并据此进行精准的预测或分类。借助已有的标记数据&#xff0c;监督学习模型学会了从输入到输出的映射关系&#x…

导弹制导方式简介-其实跟卫星定位系统关系不大

导弹制导方式其实跟卫星定位系统关系不大&#xff0c;所以所谓关闭卫星定位系统导弹就不能打是谣言&#xff01; 导弹制导是指利用不同的方式&#xff0c;选择飞行路线&#xff0c;将具有动力飞行的弹头移动一段距离之后&#xff0c;击中预先设定的目标。导弹制导系统利用其中…

【网络】十大网络协议

文章目录 1. HTTP&#xff08;HyperText Transfer Protocol&#xff0c;超文本传输协议&#xff09;2. HTTPS&#xff08;Secure Hypertext Transfer Protocol&#xff0c;安全超文本传输协议&#xff09;3. HTTP/34. TCP&#xff08;Transmission Control Protocol&#xff0c…

树莓派5-番外篇-GPU相关-学习记录2

树莓派5-番外篇-GPU相关 要查看你的树莓派5是否支持GPU计算&#xff0c;以及如何启用和使用它&#xff0c;你需要了解树莓派5的硬件配置和当前的驱动支持情况。以下是查看树莓派5的GPU支持情况的步骤。 树莓派5 GPU 支持概述 树莓派5 使用的是 Broadcom BCM2712 处理器&…

大数据新视界 --大数据大厂之Hive与大数据融合:构建强大数据仓库实战指南

&#x1f496;&#x1f496;&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎你们来到 青云交的博客&#xff01;能与你们在此邂逅&#xff0c;我满心欢喜&#xff0c;深感无比荣幸。在这个瞬息万变的时代&#xff0c;我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

Ps初步使用教程

1.画面快捷键 Ctrl鼠标滚轮&#xff1a;画面左右移动 Shift鼠标滚轮&#xff1a;画面上下快速移动&#xff08;不加Shift正常速度移动&#xff09; Alt鼠标滚轮&#xff1a;画面大小缩放 2.工具快捷键 ShiftG&#xff1a;渐变、油漆桶、3D材质施放 切换 CtrlO&#xff1a…

【Unity3D】如何用MMD4Mecanim插件将pmx格式模型转换为fbx格式模型

文章目录 概要一、下载MMD4Mecanim插件并导入U3D1.1 下载链接1.2 导入过程 二、将.pmx模型转换为.fbx模型三、其他参数设置3.1 VMD参数3.2 Animations 概要 在Unity的环境下&#xff0c;想要将.pmx格式的3D模型转换为.fbx是有可以用的插件的&#xff0c;并不需要某些教程中那么…

Vue3+TypeScript二次封装axios

安装如下 npm install axios 第一步&#xff1a;创建config配置文件&#xff0c;用于存放请求后端的ip地址&#xff0c;用于后期打包后便于修改ip地址。 注&#xff1a;typescript要求参数要有类型。&#xff08;ES6 定义对象 属性 类型 修改的是属性的值&#xff09; inte…

超级干货|AI产品经理6大知识体系,【附零基础小白入门指南】

想要转行AI产品经理的宝子&#xff0c;这6大知识体系是你入门的基础 &#x1f4a5;基础知识&#xff1a;AI产品的根基 &#x1f4a5;平台和硬件支持&#xff1a;AI产品的技术基础设施 &#x1f4a5;AI核心技术&#xff1a;推动产品创新的引擎 &#x1f4a5;行业实践应用&#…

Python计算机视觉 第8章-图像内容分类

Python计算机视觉 第8章-图像内容分类 8.1 K邻近分类法&#xff08;KNN&#xff09; 在分类方法中&#xff0c;最简单且用得最多的一种方法之一就是 KNN&#xff08;K-Nearest Neighbor ,K邻近分类法&#xff09;&#xff0c;这种算法把要分类的对象&#xff08;例如一个特征…

知网合作商AEPH出版,学生/教师均可投稿,优先录用教育社科领域,往期最快2周见刊

AEPH出版社旗下有5本学术期刊&#xff0c;专门出版自然科学、社会科学研究与教育领域论文的高影响力期刊&#xff0c;拥有正规ISSN号&#xff0c;出版类型涉及应用和理论方面的原创和未曾公开发表的研究论文&#xff0c;分配独立DOI号。AEPH作为中国知网&#xff08;CNKI&#…

COCOS:(飞机大战01)背景图无线循环向下滚动

飞机大战知识点总结 背景图宽高&#xff1a;480*852 将背景图移动到Canvas中 设置图2的Y轴为852&#xff0c;这样图1和图2就衔接上了 创建控制背景的ts文件 import { _decorator, Component, Node } from cc; const { ccclass, property } _decorator;ccclass(Bg) export cla…

HTTPS证书申请

&#x1f310; JoySSL CA机构 机构介绍&#xff1a;JoySSL是网盾安全基于全球可信顶级根创新推出的新一代https数字证书&#xff0c;也是目前为数不多的中国自主品牌SSL证书。 服务特点&#xff1a;JoySSL携手全球权威CA机构&#xff0c;全球多节点服务器验证签发&#xff0c;安…

MySQL系列—8.存储结构

目录 1.系统表空间 ibdata 2.通用表空间 .ibd 3.独立表空间 4.Undo 表空间 5.临时表空间 6.Redo Log File 1.系统表空间 ibdata 系统表空间由参数innodb_data_file_path定义路径、初始化大小、自动扩展策略 如&#xff1a; innodb_data_file_path/dayta/mysql/ibdata1:…