搜索算法(三) 回溯法

news2025/1/10 13:31:17

1.回溯法

回溯法可以理解成一种特殊的深度优先算法,比起普通的DFS,多了还原当前节点的一步。

修改当前节点、递归子节点、还原当前节点。

本质是一种试错的思想。

维基百科:

 2.例题

1)

力扣https://leetcode.cn/problems/permutations/修改和还原当前节点放在了dfs函数的开头和结尾,中间遍历未访问的节点,递归访问这些节点。

class Solution {
public:
    vector<vector<int>> permute(vector<int>& nums) {
        int n = nums.size();
        vector<bool> visit(n,false);
        vector<int> ans;
        for(int i=0;i<n;i++){
            dfs(nums,visit,i,ans);
        }

        return res;
    }

    void dfs(vector<int>& nums, vector<bool>& visit, int k, vector<int>& ans){
        visit[k] = true;
        ans.push_back(nums[k]);
        int n = nums.size();
        if(ans.size()==n){
            res.push_back(ans);
        }else{
            for(int i=0;i<n;i++){
            if(!visit[i]){
                dfs(nums,visit,i,ans);
                }
            }
        }
        visit[k] = false;
        ans.pop_back();
    }

private:
    vector<vector<int>> res;    
};

2)

力扣https://leetcode.cn/problems/combinations/这题与上一题类似,只是停止搜索的条件变为组合长度为k

class Solution {
public:
    vector<vector<int>> combine(int n, int k) {
        vector<int> ans;
        for(int i=1;i<=n;i++){
            dfs(i,n,k,ans);
        }

        return res;
    }

    void dfs(int start, int n, int k, vector<int>& ans){
        ans.push_back(start);
        if(ans.size()==k){
            res.push_back(ans);
        }else{
            for(int i=start+1;i<=n;i++){
                dfs(i,n,k,ans);
            }
        }
        ans.pop_back();
    }

private:
    vector<vector<int>> res;
};

3)

力扣https://leetcode.cn/problems/word-search/这题也是一样的回溯结构,深搜符合条件的路径

class Solution {
public:
    bool exist(vector<vector<char>>& board, string word) {
        int m = board.size();
        int n = board[0].size();
        vector<vector<bool>> visit(m,vector<bool>(n,false));
        bool res;
        for(int i=0;i<m;i++){
            for(int j=0;j<n;j++){
                if(board[i][j]==word[0]){
                    res = dfs(board,word,i,j,1,visit);
                    if(res==true){
                        return true;
                    }
                }
            }
        }
        return false;
    }

    bool dfs(vector<vector<char>>& board, string word, int a, int b, int x, vector<vector<bool>>& visit){
        if(word.size()==x) return true;
        visit[a][b] = true;
        int m = board.size();
        int n = board[0].size();
        for(int i=0;i<4;i++){
            int c = a + path[i];
            int d = b + path[i+1];
            if(c>=0 && c<m && d>=0 && d<n && visit[c][d]==false && 
                word[x]==board[c][d]){
                    if(dfs(board,word,c,d,x+1,visit)){
                        return true;
                    }
                }
        }
        visit[a][b] = false;
        return false;
    }
private:
    vector<int> path{-1,0,1,0,-1};
};

4)

力扣https://leetcode.cn/problems/n-queens/N皇后是经典的回溯问题,一行一行地放置皇后,每次放置前,判断一下是否符合每列每斜线只有有一个皇后的约束。放置皇后后,继续深搜下一行可能的放置方式,搜索完毕后,取消放置该行皇后。

class Solution {
public:
    vector<vector<string>> solveNQueens(int n) {
        vector<string> ans(n,string(n,'.'));
        dfs(0,n,ans);
        return res;
    }

    void dfs(int start, int n, vector<string>& ans){
       
        if(start==n){
            res.push_back(ans);
            return;
        }
        for(int i=0;i<n;i++){
             if(judge(ans,start,i)){
                ans[start][i] = 'Q';
                dfs(start+1, n, ans);
                ans[start][i] = '.';
             }
           
        }
    }

    bool judge(vector<string>& ans, int x, int y){
       int n = ans.size();
       for(int i=0;i<n;i++){
           if(ans[i][y]=='Q'){
               return false;
           }
       }
       for(int i=x-1, j=y-1;i>=0&&j>=0;i--,j--){
           if(ans[i][j]=='Q'){
               return false;
           }
       }
       for(int i=x-1,j=y+1;i>=0&&j<n;i--,j++){
           if(ans[i][j]=='Q'){
               return false;
           }
       }
       return true;
    }

private:
    vector<vector<string>> res;
};

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

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

相关文章

树莓派安装系统

0. 实验准备 树莓派一个&#xff0c;TF卡&#xff08;4GB以上&#xff09;一个&#xff0c;读卡器一个 1. 使用官方提供的工具 在搜索引擎中搜索树莓派&#xff08;不要用百度&#xff0c;建议使用必应的国际版进行搜索&#xff09;&#xff0c;我这里直接放上树莓派官方超链…

深入篇【Linux】学习必备:理解【Linux软件包管理器】yum + yum的具体使用 + yum下载的有趣指令

这里写目录标题 Ⅰ.Linux软件包管理器yum①.什么是软件包/什么是yum②.linux的软件生态与yum源③.关于rzsz Ⅱ.yum基本指令①.查看软件②.安装软件③.卸载软件 Ⅲ.yum下载的好玩指令①.sl②.linux_logo③.elinks Ⅰ.Linux软件包管理器yum yum 是一个 Shell 前端软件包管理器。基…

C++ 多态 最详细图文+代码讲解

感谢各位 点赞 收藏 评论 三连支持 本文章收录于专栏【C进阶】 ❀希望能对大家有所帮助❀ 本文章由 风君子吖 原创 回顾 上篇文章&#xff0c;我们学习了继承的相关知识&#xff0c;详细解刨了继承中的各种细节&#xff0c;而本章内容将在继承的基础上学习多态 多态的概念…

Dreamweaver如何进行网页开发?

文章目录 0.引言1.安装Dreamweaver2.编写第一个网页 0.引言 笔者本科学习的编程语言主要是关于桌面开发的&#xff0c;对编程有一定的基础&#xff0c;而编程除了关于桌面软件开发&#xff08;VisualStudio如何进行桌面软件开发&#xff1f;&#xff09;&#xff0c;还有手机应…

网络安全自学笔记+学习路线(超详细)

01 什么是网络安全 网络安全可以基于攻击和防御视角来分类&#xff0c;我们经常听到的 “红队”、“渗透测试” 等就是研究攻击技术&#xff0c;而“蓝队”、“安全运营”、“安全运维”则研究防御技术。 无论网络、Web、移动、桌面、云等哪个领域&#xff0c;都有攻与防两面…

c语言编程练习题:7-193 两小时学完C语言

#include <stdio.h> int main(){int n,k,m;int sum;if (scanf("%d %d %d",&n,&k,&m)!EOF){sum n-k*m;}printf("%d",sum);return 0; }

(学习日记)2023.04.28

写在前面&#xff1a; 由于时间的不足与学习的碎片化&#xff0c;写博客变得有些奢侈。 但是对于记录学习&#xff08;忘了以后能快速复习&#xff09;的渴望一天天变得强烈。 既然如此 不如以天为单位&#xff0c;以时间为顺序&#xff0c;仅仅将博客当做一个知识学习的目录&a…

基于SSM的教务管理系统的设计与实现(论文+源码)_kaic

摘 要 学校教务管理信息化是提高办公效率的主要途径。随着中国高等教育的快速发展和学校规模的不断扩大&#xff0c;在校学生人数不断增加&#xff0c;办学层次出现多元化&#xff0c;由一地办学发展到多地多点办学&#xff0c;同时进一步推行学分制。这些变化使得教务部门的管…

【AI绘图】四、stable difusion提示词如何写?

上一篇&#xff1a;【AI绘图】三、stable diffussion操作界面介绍以及如何使用 如何写好提示词&#xff1f; 写出一份比较好的提示词是文生图技术的关键。但是&#xff0c;写出一份好的prompt并不容易&#xff0c;下面针对“如何写好提示词”这个问题&#xff0c;从提示词构成…

MySQL 数据操纵语言 DML

文章目录 数据操纵语言 DMLINSERT 语句UPDATE 语句DELETE 语句 数据操纵语言 DML 数据操纵语言&#xff08;Data Manipulation Language&#xff0c;DML&#xff09;是 SQL 语言的核心部分之一。在添加、更新或者删除表中的数据时&#xff0c;需要执行 DML 语句。很多时候我们提…

Unity导入Android aar包实现交互全流程

一.搭建Android项目 1.创建一个Android空项目 点击finish后,就等待编译,过程中会自动下载一些插件 等待... 等待... 编译完成: 2.创建Module 右键该工程的app,新建一个Module, Language:Kotlin语法和Java语法,语法不同,后续创建的代码文件有所差异,但不影响代码编…

地震勘探基础(六)之地震反褶积

地震反褶积 地震资料常规处理主要包括地震反褶积&#xff0c;水平叠加和偏移成像三大内容。水平叠加可以提高地震资料的信噪比&#xff0c;偏移成像可以提高地震资料的空间分辨率和保真度&#xff0c;地震反褶积可以提高地震资料的分辨率和压制干扰波。 1954年&#xff0c;Ro…

Python心经(5)

目录 python对于类和实例&#xff0c;&#xff0c;都能随时动态绑定 属性或者函数 可以通过__slots__去限定实例所能绑定属性的范围 python里面类很多定制函数__xx__ 下面给个python里面枚举类型 对type函数去实现 有关迭代器&#xff1a; 生成器&#xff1a; 可迭代对…

【Linux编译器gcc/g++】带你了解代码是如何变成可执行程序的!

Linux编辑器gcc/g的使用 背景知识gcc的使用预处理&#xff08;进行宏替换&#xff09;编译(生成汇编)汇编(生成及其可识别代码)链接(生成可执行文件或者库文件)库 动态库vs静态库 debug和release&#x1f340;小结&#x1f340; &#x1f389;博客主页&#xff1a;小智_x0___0x…

【FPGA零基础学习之旅#6】ip核基础知识之计数器

&#x1f389;欢迎来到FPGA专栏~ip核基础知识之计数器 ☆* o(≧▽≦)o *☆嗨~我是小夏与酒&#x1f379; ✨博客主页&#xff1a;小夏与酒的博客 &#x1f388;该系列文章专栏&#xff1a;FPGA学习之旅 文章作者技术和水平有限&#xff0c;如果文中出现错误&#xff0c;希望大家…

[网络工程]小型局域网组建的常用命令(ENSP)

⭐作者介绍&#xff1a;大二本科网络工程专业在读&#xff0c;持续学习Java&#xff0c;努力输出优质文章 ⭐作者主页&#xff1a;逐梦苍穹 目录 1、引言2、常用命令(ENSP)常规VTYConsole端口安全单臂路由DHCPNATACL基础ACL高级ACL GVRPSTP 1、引言 局域网&#xff08;Local A…

C++学习——第一节课-初识C++

大家好&#xff0c;我是涵子。今天我们来开始学习C。 目录 一、课前准备 二、C的第一个程序 2.1.C是个啥 2.2.C的第一个程序编写 2.2.1.头文件 2.2.2.命名空间 2.2.3.主程序函数 2.2.4.输出流 2.2.5.代码结束 三、其它的应用 3.1.输出三角形&#xff0c;矩形和勾 …

单例模式C++实现和观察者模式C++实现

目录 1、单例模式介绍 2、单例代码实现 2.1 static介绍 2.2 C中static的三种用法&#xff1a; &#xff08;1&#xff09;静态局部变量 &#xff08;2&#xff09;静态成员变量 &#xff08;3&#xff09;静态成员函数 3、观察者模式介绍 4、观察者代码实现 1、单例模…

.ini配置文件介绍与解析库使用

【前言】 ini 文件是英文"Initialization"的缩写&#xff0c;即初始化文件。它用来配置特定应用软件以实现对程序初始化或进行参数设置。.ini文件由节(section)、键(key)、值(value)三种模块构成。在windows系统/嵌入式软件中有很多XXX.ini文件&#xff0c;例如Syste…