DFS之剪枝与优化

news2025/7/15 23:25:30

剪枝

1.优化搜索顺序:在大部分情况下,我们应该优先搜索分支较少的结点

2.排除等效冗余(在不考虑顺序的情况下,尽量用组合的方式来搜索)

3.可行性剪枝

4.最优性剪枝

5.记忆化搜索 

165. 小猫爬山 - AcWing题库

import java.util.*;

public class Main{
    static int N = 20;
    static int n, m, res = N;
    static int[] w = new int[N];//每只小猫的数量
    static int[] sum = new int[N];//每个缆车已经放的小猫的重量之和
    
    public static void dfs(int u, int k){//u猫的数量,共有k个缆车
        //最优性剪枝
        if(k >= res) return;
        if(u == n){
            res = k;
            return;
        }
        
        //能在以前的车里面找到位置
        for(int i = 0; i < k; i ++){
            if(sum[i] + w[u] <= m){//可行性剪枝
                sum[i] += w[u];
                dfs(u + 1, k);
                sum[i] -= w[u];//回溯,恢复现场
            }
        }
        
        //新开一辆车
        sum[k] = w[u];//数组是从0开始的,所以这里的第k+1辆车用k表示
        dfs(u + 1, k + 1);
        sum[k] = 0;//恢复现场
    }
    
    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);
        n = sc.nextInt();
        m = sc.nextInt();
        for(int i = 0; i < n; i ++){
            w[i] = sc.nextInt();
        }
        
        //优化搜索顺序,分支少的先搜索(重猫)
        Arrays.sort(w, 0, n);//排序
        int[] b = new int[n];
        for(int i = n - 1, j = 0; i >= 0; i --){
            b[j ++] = w[i];
        }
        for(int i = 0; i < n; i ++) w[i] = b[i];
        
        dfs(0, 0);
        
        System.out.print(res);
    }
}

 

166. 数独 - AcWing题库

可以用一个九位的二进制数来表示一行的状态 

import java.util.*;

public class Main{
    static int N = 9, cnt;
    static char[] str;//输入输出
    static int[] row = new int[N];//行
    static int[] cel = new int[N];//列
    static int[][] cell = new int[3][3];//九宫格
    static int[] ones = new int[1 << N];//二进制状态中1的个数
    //二进制数中最后一个1的位置
    static Map<Integer, Integer> map = new HashMap<>();

    //初始化所有行列和九宫格都可以填1到9中的数
    public static void init(){
        for(int i = 0; i < N; i ++) row[i] = cel[i] = (1 << N) - 1;

        for(int i = 0; i < 3; i ++){
            for(int j = 0; j < 3; j ++){
                cell[i][j] = (1 << N) - 1;
            }
        }
    }

    //st为true表示在这个位置上填上一个t,在这个位置上去除t
    public static void draw(int x, int y, int t, boolean st){
        if(st) str[x * N + y] = (char)(t + '1');
        else str[x * N + y] = '.';

        int v = 1 << t;//第几位的1
        if(!st) v = -v;//回溯

        row[x] -= v;
        cel[y] -= v;
        cell[x / 3][y / 3] -= v;
    }

    //lowbit函数求二进制数的最后一个1
    public static int lowbit(int x){
        return x & -x;
    }

    //获取他能够放的所有的状态为1的位置
    public static int get(int x, int y){
        return row[x] & cel[y] & cell[x / 3][y / 3];
    }

    public static boolean dfs(int cnt){
        if(cnt == 0) return true;//如果空点已经没有了

        int minv = 10;//能填的数的个数
        int x = -1, y = -1;//初始化这个点的横纵坐标
        //找到能填的数最少的点
        for(int i = 0; i < N; i ++){
            for(int j = 0; j < N; j ++){
                if(str[i * N + j] == '.'){//空着的点才能填
                    int f = get(i, j);//获取它能够放的状态
                    if(ones[f] < minv){
                        minv = ones[f];
                        x = i;
                        y = j;
                    }
                }
            }
        }
        int state = get(x, y);//找到分支最小的那一点
        for(int i = state; i != 0; i -= lowbit(i)){
            int t = map.get(lowbit(i));
            draw(x, y, t, true);//进行放置
            if(dfs(cnt - 1)) return true;
            draw(x, y, t, false);//回溯
        }

        return false;//不行就返回false
    }

    //main函数
    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);

        //获取每个状态中1的个数
        for(int i = 0; i < 1 << N; i ++){
            for(int j = 0; j < N; j ++){
                ones[i] += i >> j & 1;
            }
        }

        //二进制数中最后一个1的位置
        for(int i = 0; i < N; i ++) map.put(1 << i, i);

        while(true){
            String s = sc.next();//输入数据
            if(s.equals("end")) break;
            str = s.toCharArray();//转化为数组

            init();//初始化

            cnt = 81;//一共有81个空位要填
            for(int i = 0; i < N; i ++){
                for(int j = 0; j < N; j ++){
                    //转化为二维数组来看
                    if(str[i * N + j] != '.'){
                        draw(i, j, str[i * N + j] - '1', true);//填上这个数
                        cnt --;//减去空着的点数
                    }
                }
            }

            dfs(cnt);

            //把char数组转化为字符串输出
            System.out.println(new String(str));
        }
    }
}

 

167. 木棒 - AcWing题库 

import java.util.*;

public class Main{
    static int N = 70;
    static int length, sum, n;
    static int[] w = new int[N];//每根小棍的长度
    static boolean[] st = new boolean[N];//标记这个小棍用过了没有
    
    public static boolean dfs(int u, int s, int start){//第几根大棍,这个大棍的长度,从第几根小棍开始枚举
        if(u * length == sum) return true;
        if(s == length) return dfs(u + 1, 0, 0);//如果当前大棍已经满了
        
        for(int i = start; i < n; i ++){
            if(st[i]) continue;//已经用过了
            //可行性剪枝
            if(s + w[i] > length) continue;//超过大小了
            
            st[i] = true;//标记一下
            if(dfs(u, s + w[i], i + 1)) return true;
            st[i] = false;//恢复现场
            
            if(s == 0) return false;//放在第一个位置失败
            if(s + w[i] == length) return false;//放在最后一个位置失败
            
            int j = i;
            while(j < n && w[i] == w[j]) j ++;
            i = j - 1;
        }
        return false;
    }
    
    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);
        while(true){
            length = 0;
            sum = 0;
            Arrays.fill(st, false);//多组测试数据
            
            n = sc.nextInt();
            if(n == 0) break;
            for(int i = 0; i < n; i ++){
                w[i] = sc.nextInt();
                length = Math.max(length, w[i]);
                sum += w[i];//所有大棍的总长度
            }
            
            //从大到小排序,优化搜索顺序
            Arrays.sort(w, 0, n);
            int[] b = new int[n];
            for(int i = n - 1, j = 0; i >= 0; i --){
                b[j ++] = w[i];
            }
            for(int i = 0; i < n; i ++){
                w[i] = b[i];
            }
            
            while(true){
                //判断总长度是不是length的倍数
                if(sum % length == 0 && dfs(0, 0, 0)){
                    System.out.println(length);//因为length从小到大枚举,所以第一次成立的时候就是最小值
                    break;
                }
                length ++;
            }
        }
    }
}

 

168. 生日蛋糕 - AcWing题库

 

还不是太懂,先跳一下 

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

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

相关文章

【快速上手QT】06-检测按键检测鼠标

今天聊聊事件&#xff0c;实际上我们在前两篇文章中就已经接触到了事件&#xff0c;分别是定时器事件和绘画事件&#xff0c;今天我们再来看看其他的事件。 我们打开QT助手&#xff0c;在QWidget的界面中找到下图中的地方。 我们可以看到很多函数的结尾都是Event&#xff0c;那…

日本极致产品力|200人的小型家族企业,如何年销7亿块巧克力?

蒂罗尔巧克力是日本经典的巧克力品牌。糖果业务是其早期的主营业务&#xff0c;在主营业务下滑的情况下&#xff0c;确立新的竞争方向、打造新产品、寻找新方法&#xff0c;成就巧克力极致产品力重回增长。 竞争环境变化&#xff1a;糖果主营业务持续下滑 始于1903年的松尾株式…

云原生架构技术揭秘:探索容器技术的奥秘

云原生的概念和演进都是围绕云计算的核心价值展开的&#xff0c;比如弹性、自动化、韧性&#xff0c;所以云原生所涵盖的技术领域非常丰富。 随着云计算技术的不断发展&#xff0c;云原生架构已经成为了新一代软件开发的重要趋势。本文将为您介绍云原生架构的相关技术&#xf…

产品经理岗位的任职资格和职业规划

产品经理主要是商业银行以客户为导向的&#xff0c;具体负责组织银行某一金融产品线的创新设计、生产营销和管理服务的工作。这类人士主要负责应用实施工作&#xff0c;其中产品线由一系列的产品构成&#xff0c;公司的产品经理主要分为全过程产品创新设计专家、全过程产品生产…

Decision Transformer

DT个人理解 emmm, 这里的Transformer 就和最近接触到的whisper一样,比起传统Transformer,自己还设计了针对特殊情况的tokens。比如whisper里对SOT,起始时间,语言种类等都指定了特殊tokens去做Decoder的输入和输出。 DT这里的作为输入的Tokens由RL里喜闻乐见的历史数据:…

QtCreator报Failed to parse qmlimportscanner output解决

错误如下: 定位错误位置 增加错误信息打印 打印执行命令 执行打印输出的命令,成功返回JSON 但输出的JSON对象不是json格式,而是命令 增加$$成功输出JSON 使用QtCreator12编译一次后,再使用QtCreator13成功编译通过,问题解决

Floyd算法、Dijkstra算法、基础拓扑排序

Floyd算法 Dijkstra算法 基础拓扑排序

TikTok运营应该使用什么IP?网络问题大全

想要迈过TikTok新手门槛&#xff0c;首先必须要学习的就是网络问题。很多人开始做TikTok账号或者TikTok小店时&#xff0c;都会遇到一些先前没有遇到的词汇和概念&#xff0c;比如原生IP&#xff0c;独享IP&#xff0c;甚至专线&#xff0c;那么一个IP可以做几个账号呢&#xf…

编译 qsqlmysql.dll QMYSQL driver not loaded

Qt 连接MySQL数据库&#xff0c;没有匹配的qsqlmysql.dll, 需要我们跟进自己Mysql 以及QT版本自行编译的。异常如下图&#xff1a; 安装环境为 VS2019 Qt5.12.12&#xff08;msvc2017_64、以及源码&#xff09; 我的安装地址&#xff1a;D:\Qt\Qt5.12.12 Mysql 8.1.0 默认安…

【C++从0到王者】第四十五站:图

文章目录 一、图的概念1.图概念2.顶点与边的概念3.有向图和无向图4.完全图5.邻接顶点6.顶点的度7.路径与路径长度8.简单路径与回路9.子图10.连通图与强连通图11.生成树 二、图的存储结构1.邻接矩阵1.1 基本概念1.2 代码实现 2.邻接表1.1 基本概念1.2 代码实现 3.总结 一、图的概…

如何设计一个秒杀系统?

秒杀是电商系统中常见的业务&#xff0c;用于吸引用户&#xff0c;刺激留存及消费所做的一种活动。经典的秒杀包含限时秒杀和限量秒杀。很多公司有专门的秒杀系统。哪个业务要做活动&#xff0c;就来对接这个系统。 系统特点 1、瞬时流量极大&#xff0c;过了秒杀时间点流量结束…

【HTML】HTML基础2(一些常用标签)

目录 例子 首先是网页图标 然后是一些常用标签 插入图片 例子 <!DOCTYPE html> <html><head><link rel"icon" href"img/银河护卫队-星爵.png" type"image/x-icon"><meta charset"utf-8"><title>…

[RoarCTF 2019]Easy Calc

这题考查的是: 字符串解析特性目录读取文件内容读取 字符串解析特性详解&#xff1a;PHP字符串解析特性 &#xff08;$GET/$POST参数绕过&#xff09;&#xff08;含例题 buuctf easycalc&#xff09;_参数解析 绕过-CSDN博客 ascii码查询表&#xff1a;ASCII 表 | 菜鸟工具 …

人工智能_大模型010_Centos7.9中CPU安装ChatGLM3-6B大模型_安装使用_010---人工智能工作笔记0145

从一个空的虚拟机开始安装: https://www.modelscope.cn/models/ZhipuAI/chatglm3-6b/files 可以看到这里有很多的数据文件,那么这里 这里点击模型文件就可以下载,这个就是chatglm3-6B的文件,需要点击每个文件,然后点击右边的下载,把文件都下载下来 右侧有下载按钮.点击下载可…

如何使用固定公网地址远程访问本地RStudio Server【内网穿透】

文章目录 前言1. 安装RStudio Server2. 本地访问3. Linux 安装cpolar4. 配置RStudio server公网访问地址5. 公网远程访问RStudio6. 固定RStudio公网地址 前言 RStudio Server 使你能够在 Linux 服务器上运行你所熟悉和喜爱的 RStudio IDE&#xff0c;并通过 Web 浏览器进行访问…

操作系统系列学习——系统调用的实现

文章目录 前言系统调用的实现 前言 一个本硕双非的小菜鸡&#xff0c;备战24年秋招&#xff0c;计划学习操作系统并完成6.0S81&#xff0c;加油&#xff01; 本文总结自B站【哈工大】操作系统 李治军&#xff08;全32讲&#xff09; 老师课程讲的非常好&#xff0c;感谢 【哈工…

(C语言)二分查找

在⼀个升序的数组中查找指定的数字n&#xff0c;很容易想到的⽅法就是遍历数组&#xff0c;但是这种⽅法效率⽐较低。⽐如我买了⼀双鞋&#xff0c;你好奇问我多少钱&#xff0c;我说不超过300元。你还是好奇&#xff0c;你想知道到底多少&#xff0c;我就让你猜&#xff0c;你…

定时任务调动框架Quartz+SpringBoot集成

Quartz 是一个基于 Java 的广泛使用的开源的任务调度框架 官网&#xff1a; http://www.quartz-scheduler.org/ 源码&#xff1a; https://github.com/quartz-scheduler/quartz 整个 Quartz 的代码流程基本基本如下&#xff1a; 1、首先需要创建我们的任务(Job)&#xff0c…

【精品】集合list去重

示例一&#xff1a;对于简单类型&#xff0c;比如String public static void main(String[] args) {List<String> list new ArrayList< >();list.add("aaa");list.add("bbb");list.add("bbb");list.add("ccc");list.add(…

Python进阶学习:Pandas--向DataFrame中指定位置添加一列或多列

Python进阶学习&#xff1a;Pandas–向DataFrame中指定位置添加一列或多列 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f44…