算法提高-搜索-DFS之剪枝与优化

news2024/11/19 19:18:47

DFS之剪枝与优化

  • DFS之剪枝与优化
    • AcWing 165. 小猫爬山
    • AcWing 166. 数独
    • AcWing 167. 木棒
    • AcWing 168. 生日蛋糕

DFS之剪枝与优化

AcWing 165. 小猫爬山

DFS的五种剪枝方法
(1)优化搜索顺序
(2)排除等效冗余
(3)可行性剪枝
(4)最优性剪枝
(5)记忆化搜索(这个放在dp里面说了)
下面这题涉及三个剪枝

#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;

const int N = 20;

int n;
int maxW;
int sum[N];
int w[N];
int ans = N;//这里debug很久,题目找的是最小值,初始化的时候要给他定义一个最大值

void dfs(int u, int k)
{
    // (3)最优性剪枝
    if (k >= ans) return ;
    if (u == n)
    {
        ans = k;
        return ;
    }
    //1.判断能否加入现在已经存在的缆车
    //遍历缆车
    for (int i = 0; i < k; i ++ )
    {
        // (2)可行性剪枝
        if (sum[i] + w[u] <= maxW)
        {
            sum[i] += w[u];
            dfs(u + 1, k);
            sum[i] -= w[u];//回溯
        }
    }
    //2.判断是否需要新开一个缆车
    sum[k] = w[u];//因为我们缆车的下标是从0开始的
    dfs(u + 1, k + 1);
    //sum[k] = 0;   sum是全局变量,但这不是sum += w[u],其实不用回溯,我们返回到k - 1层的时候会直接重新赋值覆盖sum[k]
}

int main()
{
    cin >> n >> maxW;
    for (int i = 0; i < n; i ++ ) cin >> w[i];
    // (1)优化搜索顺序
    sort(w, w + n);
    reverse(w, w + n);
    dfs(0, 0);
    cout << ans;
    return 0;
}

AcWing 166. 数独

AcWing 167. 木棒

6种剪枝方法:1, 2, 3-1, 3-2, 3-3, 3-4
在这里插入图片描述

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;
const int N = 64 + 5;

int sum;//所有小棒加起来的总长度,也是所有大棒加起来的总长度
int length;//枚举每一根大棒的长度
int n;
int stick[N];//记录每根木棍的长度
int st[N];


bool dfs(int u, int cur, int start) 
{
    if (u * length == sum) return true;//找到一组满足题意的解
    if (cur == length) return dfs(u + 1, 0, 0);//第u根木棒组合完毕,枚举第u+1根,直到u * length == sum才代表满足题意

    for (int i = start; i < n; i ++ )
    {
        if (st[i]) continue;
        if (stick[i] + cur > length) continue;//可行性剪枝
        
        st[i] = true;
        if (dfs(u, cur + stick[i], i + 1)) return true;//这一步相当编译器帮我们回溯
        st[i] = false;//手动回溯
        
        //剪枝 3-3 和 3-4,
        //走到了这一步说明之前dfs return false了,我们判断一下是将小棍放在大棍的哪个位置失败了
        //根据3-3和3-4,如果是放在大棍的第一个位置和放在大棍的最后一个位置失败了,那么就直接不同继续判断下去了
        //当前这个小棍stick[i]也一定不能放在后续的其他大棍中,必找不到一组合法解
        if (cur == 0 || cur + stick[i] == length) return false;
        
        //剪枝3-2如果当前的小棍失败了,那么略过枚举stick[]中后面和它长度一样的小棍
        int j = i + 1;
        while (j < n && stick[j] == stick[i]) j ++ ;
        i = j - 1;//上面的while能退出说明stick[i]已经!=stick[j]了,所有要 -1
    }
    return false;
}



int main ()
{
    while (cin >> n, n != 0)
    {
        //多组数据,sum和st要置0
        memset(st, 0, sizeof st);
        sum = 0;
        for (int i = 0; i < n; i ++ )
        {
            cin >> stick[i];
            sum += stick[i];
        }
        
        sort(stick, stick + n);//优化搜索顺序,先枚举长的木棒,这样后面需要搜索的分支可以减少些
        reverse(stick, stick + n);
        
        length = 1;
        while (true)
        {
            if (sum % length == 0 && dfs(0, 0, 0))
            {
                cout << length << endl; //题目求的是原始木棒的最小长度,因此我们length从1开始枚举即可,只要找到一组合法的解就break
                break;
            }
            length ++ ;//走到这一步了说明之前没break没找到一组合法的解,那么继续枚举
        }
    }
    return 0;
}

AcWing 168. 生日蛋糕

这题挺难的,我反正糊着写完了建议配合y总讲解和这篇题解复习
在这里插入图片描述

#include <cstring>
#include <iostream>
#include <algorithm>
#include <cmath>

using namespace std;

const int M = 25, INF = 1e9;
int n, m;
int R[M], H[M];//记录每一层的半径和高度
int minv[M], mins[M];//记录前i层的最小体积和,最小面积和
int ans = INF;

void dfs(int u, int v, int s)
{
    //可行性剪枝
    if (v + minv[u] > n) return ;
    if (s + mins[u] >= ans) return ;
    //这个是用公式推出来的1-u层蛋糕的面积的最小值
    if (s + 2 * (n - v) / R[u + 1] >= ans) return ;
    
    if (u == 0)//倒着枚举的,所以在第0层收集结果
    {
        if (v == n) ans = s;//前面没return,说明这里的s一定小于ans,直接赋值就行了,我们这里的ans一定是最小的
        return ;//这个return不能放在if里面,因为也许u == 0的时候v!=n,return要是放在里面如果无解就会TLE
    }
    
    //优化搜索顺序,r是二次方的,先搜r再搜h,分支会少
    for (int r = min(R[u + 1] - 1, (int)sqrt(n - v)); r >= u; r -- )//min函数一定要都是int,所以要强转
        for (int h = min(H[u + 1] -1, n - v / r / r); h >= u; h -- )//倒着枚举的要--
        {
            //特判一下,如果是m层我们直接加上整个上表面积就行,后面再枚举每个侧表面积
            int t = 0;
            if (u == m) t = r * r;
            R[u] = r, H[u] = h;
            dfs(u - 1, v + r * r * h, s + 2 * r * h + t);//2 * r * h是这一层的侧面积,t是特判出来的上表面积,就是一整个圆盘,一组解只要加1次就行
        }
}


int main ()
{
    cin >> n >> m;
    
    //预处理
    for (int i = 1; i <= m; i ++ )
    {
        minv[i] = minv[i] + i * i * i;//r * r * h
        mins[i] = mins[i] + 2 * i * i;//2 * r * r ,题目说了方便计算pi省略
    }
    //m+1层不存在,作为哨兵,减少边界情况的判断
    R[m + 1] = H[m + 1] = INF;//不加这个会报Float Point Exception,但这个y总说是除0了
    
    //优化搜索顺序,先从最大的一层搜,分支会少
    dfs(m, 0, 0);
    
    if (ans == INF) ans = 0;//特判无解的情况
    cout << ans;
    return 0;
}

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

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

相关文章

chatgpt赋能python:Python删除目录:如何在项目中正确删除文件夹?

Python 删除目录&#xff1a;如何在项目中正确删除文件夹&#xff1f; 在使用Python语言进行程序开发的过程中&#xff0c;可能会需要删除不再需要的目录&#xff0c;例如缓存和日志文件夹。然而&#xff0c;删除目录需要小心谨慎&#xff0c;避免误删除重要文件或目录。在本文…

chatgpt赋能python:Python列表自动排序

Python列表自动排序 Python是一种流行的编程语言&#xff0c;它有很多有用的内置函数和数据结构。其中一个最常用的数据结构是列表。在Python中&#xff0c;列表是一个有序的集合&#xff0c;可以存储多种类型的元素。列表不仅仅是一个数据结构&#xff0c;它还有一些有用的方…

Flink standalone 集群会话模式部署搭建

环境准备 1、Centos7集群环境搭建 2、flink-1.17.0-bin-scala_2.12.tgz 安装包&#xff0c;下载地址 规划 服务器角色iphadoop01JobManager TaskManager192.168.140.132hadoop02TaskManager192.168.140.133hadoop03TaskManager192.168.140.134 安装 1、下载 flink-1.17.0-…

【Unity3D】边缘检测特效

1 边缘检测原理 边缘检测的原理是&#xff1a;检测每个像素周围的像素亮度差&#xff0c;如果亮度差异较大&#xff0c;就将该像素识别为边缘&#xff0c;并进行边缘着色。 使用过卷积神经网络&#xff08;CNN&#xff09;的读者&#xff0c;一定知道卷积运算&#xff0c;笔者之…

储能基础知识【一】

储能基础知识【一】 1、基础名词、概念、对应的英文单词、系统组成2、储能电池系统组成图3、性能指标 1、基础名词、概念、对应的英文单词、系统组成 电池储能系统&#xff08;Battery Energy Storage System, BESS&#xff09;&#xff1b;电芯&#xff08;Battery Cell&…

总结887

学习目标&#xff1a; 周目标&#xff1a;强化强3讲&#xff0c;英语背3篇文章并回诵&#xff0c;检测&#xff0c;一套数学模拟题 每日必复习&#xff08;5分钟&#xff09; 复习第四讲方程组 学习内容&#xff1a; 暴力英语&#xff1a;背诵《happiness is a journey》每日…

《HTTPS协议原理》

【一】https协议是啥子&#xff1f; https也是一个应用层协议&#xff0c;实在http协议的基础上&#xff0c;引入了一个加密层&#xff0c;http协议的内容都是按照文本的方式进行明文传输的&#xff0c; 这就导致了在传输的过程中出现一些被篡改的情况。 【二】啥是加密&…

操作系统复习3.1.0-内存

内存 程序是由内存放到CPU才可处理&#xff0c;前面一直有提到外存、内存&#xff0c;外存I/O速度十分慢&#xff0c;而内存I/O速度快&#xff0c;CPU I/O速度也快。 因此内存是缓和外存和CPU间I/O速率差异问题 为区分并发环境下程序数据存放地方&#xff0c;就给内存的存储单…

前端 js 栈内存和堆内存 基本数据类型和复杂数据类型的区别?

前端 js 栈内存和堆内存 基本数据类型和复杂数据类型的区别&#xff1f; 先了解一下JavaScript 数据类型有哪些&#xff1f; javaScript 中有8种基本的数据类型&#xff1a;7种为基本数据类型&#xff0c;而Object 为复杂数据类型 基本数据类型&#xff08;原始数据类型&#…

OpenMMLab-AI实战营第二期-人体关键点检测与MMPose

人体关键点检测与MMPose 课程链接&#xff1a;https://www.bilibili.com/video/BV1kk4y1L7Xb 这个课程的大致内容是介绍如何从给定的二维影像中恢复出人体的姿态&#xff08;2D或者3D&#xff09;&#xff0c;大纲如下所示&#xff0c;基本上可以认为流程是&#xff1a;先是恢…

Spring Boot 日志配置(Slf4j)

SLF4J与Logback简介 Java日志框架众多&#xff0c;常用的有java.util.logging, log4j, logback&#xff0c;commons-logging等。 SLF4J (Simple Logging Facade For Java)&#xff0c;它是一个针对于各类Java日志框架的统一Facade抽象。SLF4J定义了统一的日志抽象接口&#x…

Linux命令学习之pwd和ls

pwd pwd是查看当前所在目录的命令。 可以看到当前所在目录是/root。在这里需要注意/是根目录&#xff0c;是所有其他目录的父节点&#xff0c;/root目录是root用户的家&#xff08;home&#xff09;目录&#xff0c;这是两个不同的目录。 man pwd可以看一下pwd的帮助&#xf…

Power BI 如何生成动态指标散点图

前言 本文介绍如何在Power BI中创建动态散点图&#xff0c;可以自由切换X轴和Y轴的指标。 数据下载&#xff1a; 使用的是CSDN后台的单篇文章分析数据&#xff0c;在“作品数据”页点击“导出数据”。 因为都是累计值&#xff0c;所以用了两天的数据&#xff0c;手动添加…

[数据挖掘02] pandas的分配和聚合函数

一 说明 窗口函数是什么&#xff1f;窗口函数是时间序列的局部属性处理函数&#xff0c;比如&#xff0c;一维卷积滤波、移动平均、指数平均本篇我们将针对pandas对象的窗口函数展开讨论&#xff0c;并以示例展示他们的概念实质。 二 窗口函数、分组函数&#xff08; GroupBy …

大语言模型速查表;ChatGPT发展路线图;11条市场营销ChatGPT Prompt;使用Midjourney制作专属头像 | ShowMeAI日报

&#x1f440;日报&周刊合集 | &#x1f3a1;生产力工具与行业应用大全 | &#x1f9e1; 点赞关注评论拜托啦&#xff01; &#x1f916; 大语言模型速查表 Large Language Model Cheat Sheet ShowMeAI知识星球资源编码&#xff1a;R115 本份速查表的制作目的&#xff0c;是…

Linux4.2LAMP

文章目录 计算机系统5G云计算第一章 LINUX LAMP一、概述二、编译安装Apache httpd服务1.关闭防火墙&#xff0c;将安装Apache所需软件包传到/opt目录下2.安装环境依赖包3.配置软件模块4.编译及安装5.优化配置文件路径&#xff0c;并把httpd服务的可执行程序文件放入路径环境变量…

LInux-文本处理相关命令笔记

目录 文本处理相关命令正则表达式介绍BRE和ERE seqxargs常用选项常用使用方法 uniq介绍常见选项常见使用方法实例 tr常用选项常见使用方法能转换的原因 grep常见选项grep -v 选项 -v和[^..]的区别 常见使用方法使用 总结 cut介绍使用 sort介绍语法参数 使用去重 排序原则按字典…

HarmoneyOS入门--下载与安装DevEco Studio运行helloworld

下载与安装DevEco Studio 在HarmonyOS应用开发学习之前&#xff0c;需要进行一些准备工作&#xff0c;首先需要完成开发工具DevEco Studio的下载与安装以及环境配置。 下载DevEco Studio 下载完成后&#xff0c;双击下载的“deveco-studio-xxxx.exe”&#xff0c;进入DevEco S…

chatgpt赋能python:Python创建空变量的方法

Python创建空变量的方法 Python是一种非常受欢迎的编程语言&#xff0c;因为它易于学习和使用&#xff0c;并且具有动态语言的特点。不管你是新手还是有经验的开发人员&#xff0c;你肯定会经常遇到需要创建空变量的情况。在这篇文章中&#xff0c;我们将探讨Python中创建空变…

AI对话交互场景使用WebSocket建立H5客户端和服务端的信息实时双向通信

WebSocket使得客户端和服务器之间的数据交换变得更加简单&#xff0c;允许服务端主动向客户端推送数据。在WebSocket API中&#xff0c;浏览器和服务器只需要完成一次握手&#xff0c;两者之间就可以创建持久性的连接&#xff0c;并进行双向数据传输。 一、为什么需要 WebSock…