【算法手札】深入理解宽度遍历(bfs)和深度遍历(dfs)搜索

news2025/1/12 0:47:30

   算法的重要性不言而喻,现在我们的生活也已经离不开各种算法,一个好的算法能大大提高程序的运行效率,是学习编程的一个重要模块,而遍历算法也是算法里的一个大的模块,今天我们一起来学习一下深度遍历算法(depth first search)和 广度优先遍历(broad first search)。我们通过几道经典的分析例题来理解这两种算法。

在这里插入图片描述

文章目录

  • 一、宽度遍历搜索(bfs)
      • 1.概念和算法思想
      • 2.红与黑(牛客网)
      • 3.走迷宫(牛客网)
  • 二、深度搜索搜索(dfs)
      • 1.概念和算法思想
      • 2.排列数字(acwing)
      • 3.红与黑(牛客网)(dfs)
      • 4、n皇后问题
  • 三、总结

提示:以下是本篇文章正文内容,下面案例可供参考

一、宽度遍历搜索(bfs)

1.概念和算法思想

宽度优先搜索算法(又称广度优先搜索)是最简便的图的搜索算法之一,这一算法也是很多重要的图的算法的原型。Dijkstra单源最短路径算法和Prim最小生成树算法都采用了和宽度优先搜索类似的思想。其别名又叫BFS,属于一种盲目搜寻法,目的是系统地展开并检查图中的所有节点,以找寻结果。换句话说,它并不考虑结果的可能位置,彻底地搜索整张图,直到找到结果为止。

2.红与黑(牛客网)

2.1题目链接红与黑

2.2分析题面
在这里插入图片描述
在这里插入图片描述
题意:输入对应的两个数m,和n,m是行, n是列,输入一个char类型的图,遍历这个图,找到‘@‘这个点,并且从’@‘点开始计算能走到的’.'字符的总个数。
算法分析:flood fill (大水漫灌算法)
bfs算法适合做最短路的问题,遍历顺序,从’@'这个点往上右下左四个方向去搜索,所以我们可以设置两个方向数组,就可以实现。bfs的实现还要依靠队列来完成,一般是默认是STL库里提供的队列就可以了。
在这里插入图片描述
这道题的实现思路:
在这里插入图片描述
代码示例:

#include<iostream>
#include<queue>
#include<algorithm>
using namespace std;
typedef pair<int, int> pii;
#define x first //这样子代码会比较短
#define y second 
const int N = 25;
char g[N][N]; //存地图
int st[N][N];
int m, n; //m是列, n是行
int bfs(int sx, int sy)
{
    queue<pii> q; // 创造一个pii类型的队列,因为是二维数组
    q.push({sx, sy});
    st[sx][sy] = '#';// 标记为为障碍物
    int res = 0; //计算总共能走的数有多少
    int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
    while(q.size())
    {
        auto t = q.front();
        q.pop();
        res++;
        for(int i = 0; i < 4; i++)
        {
            int a = t.x + dx[i], b = t.y + dy[i]; //这个是新的点
            if(a < 0 || a >= n || b < 0 || b >= m || g[a][b] != '.' ) continue;
            g[a][b] ='#';
            q.push({a, b});
        }
    }
    return res;
}

int main ()
{
    while(cin >> m >> n, n || m)
    {
        for(int i = 0; i < n; i++) cin >> g[i];  //这样就把二维数组存进来了
        int a, b;
        for(int i = 0; i < n; i++)
            for(int j =  0; j < m; j ++)
            if(g[i][j] == '@') //找到@这个点的下标
            {
                a = i;
                b = j;
            }
        cout << bfs(a, b) << endl;
    }
    
    return 0;
}

3.走迷宫(牛客网)

3.1题目链接走迷宫

3.2分析题面
在这里插入图片描述
题意:入口固定在第一行第二列,出口固定在最后一行第九列,其中’#‘代表墙, '.'代表通路,算从入口到出口的最少步数u,就相当于是最短路,也能用bfs解。
算法分析:flood fill (大水漫灌算法)
解法与上题基本没区别,但是要注意bfs函数的实现。

在这里插入图片描述
代码示例:

#include<iostream>
#include<queue>
#include<cstring>
#include<algorithm>
using namespace std;
typedef pair<int, int> PII;
#define x first
#define y second
const int N = 11;
char g[N][N]; //存图
int d[N][N]; //存路径


int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0 ,-1};
int bfs()
{
    queue<PII> q;
    memset(d, -1, sizeof d);
    d[0][1] = 0;
    q.push({0, 1});
    while(q.size())
    {
        auto t = q.front();//取对头
        q.pop();
        for(int i = 0; i < 4; i++)
        {
            int a = t.x + dx[i], b = t.y + dy[i];
            if(a >= 0 && a < 10 && b < 10 && b >= 0 && g[a][b] == '.' && d[a][b] == -1)
            {
                d[a][b] = d[t.x][t.y] + 1;
                q.push({a, b});
            }
        }
    }
    
    return d[9][8];
}

int main ()
{
    while(cin >> g[0][0])
    {
        for(int i = 0; i < 10; i++)
        {
              for(int j = 0; j < 10; j++)
            {
                if(i == 0 && j == 0) continue;
                cin >> g[i][j];
            }
        }
            cout << bfs() << endl;
    }   
    
    return 0;
}

二、深度搜索搜索(dfs)

1.概念和算法思想

概念:事实上,深度优先搜索属于图算法的一种,英文缩写为DFS即Depth First Search.其过程简要来说是对每一个可能的分支路径深入到不能再深入为止,而且每个节点只能访问一次。
算法思想:
深度优先遍历图的方法是,从图中某顶点v出发:
(1)访问顶点v;
(2)依次从v的未被访问的邻接点出发,对图进行深度优先遍历;直至图中和v有路径相通的顶点都被访问;
(3)若此时图中尚有顶点未被访问,则从一个未被访问的顶点出发,重新进行深度优先遍历,直到图中所有顶点均被访问过为止。 当然,当人们刚刚掌握深度优先搜索的时候常常用它来走迷宫.事实上我们还有别的方法,那就是广度优先搜索(BFS).

2.排列数字(acwing)

2.1分析题面
在这里插入图片描述
给定一个整数n,给出1 - n的全排列。
算法分析: dfs算法,顺序很重要
在这里插入图片描述
代码示例:

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

const int N = 10;
int p[N]; //存每一次的数
bool st[N]; //标记是否走过
int n; //n设置为全局会更好
int dfs(int u)
{
    if(u == n) //如果u的大小等于n,此时就可以输出p数组
    {
        for(int i = 0; i < n; i++) printf("%d ", p[i]);
        puts("");
    }
    for(int i =  1; i <= n; i++)
    {
        if(!st[i])//如果i这个点没有被标记过,就可以进来
        {
            p[u] = i;  //p[u] 这个点i
            st[i] = true; //标记
            dfs(u + 1); //继续往深处递归
            st[i] = false; //恢复现场
        } 
    }
    
}
int main ()
{
    cin >> n;
    
    dfs(0);
    
    return 0;
}

3.红与黑(牛客网)(dfs)

3.1分析题面
在这里插入图片描述
在这里插入图片描述
题意: 输入对应的两个数m,和n,m是行, n是列,输入一个char类型的图,遍历这个图,找到‘@‘这个点,并且从’@‘点开始计算能走到的’.'字符的总个数。
3.2算法分析
在这里插入图片描述

代码示例:

#include<iostream>
using namespace std;
const int N = 25;
int n, m;
char g[N][N];
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};//枚举四个方向
int dfs(int x, int y)
{
    int res = 1;
    g[x][y] = '#';//表示这个点已经走过了
    for(int i = 0; i < 4; i++)
    {
        int a = x + dx[i], b = y + dy[i];
        if(a < 0 || a >= n || b < 0 || b >= m || g[a][b] == '#') continue;//枚举边界条件
        res += dfs(a, b);
    }
    return res;
}
int main ()
{
    while(cin >> m >> n, n || m)
    {
        for(int i = 0; i < n; i++) cin >> g[i];
        int x, y;
        
        for(int i = 0; i < n; i++)
            for(int j = 0; j < m; j++)
            if(g[i][j] == '@')
            {
                x = i;
                y = j;
            }
        cout << dfs(x, y) << endl;
    }
    
    return 0;
}

4、n皇后问题

4.1分析题面在这里插入图片描述
在这里插入图片描述
n个皇后放在n x n的国际象棋上,就是每一行要放一个皇后,但是任意一个皇后不能处在同一行和同一列和同一斜线上。

4.2算法分析(dfs)
使用类似全排列的方式,把符合条件的选出来。
在这里插入图片描述

代码示例:

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

const int N = 20;
char g[N][N];//存图
bool col[N], zx[N], yx[N];//col代表行, zx代表左斜线。yx代表右斜线
int n;
void dfs(int u)
{
    if(n == u)//如果n == u表示此时g数组里已经有结果了
    {
        for(int i = 0; i < n; i++) puts(g[i]);
        puts("");
        return;
    }
    for(int i = 0; i < n; i++)
    {
        if(!col[i] && !zx[u + i] && !yx[n - u + i])//如果行,左斜线,右斜线都没被标记过
        {
            g[u][i] = 'Q';
            col[i] = zx[u + i] = yx[n - u + i] = true;
            dfs(u + 1);
            col[i] = zx[u + i] = yx[n - u + i] = false;
            g[u][i] = '.'; //恢复现场
        }
    }
}
int main ()
{
    cin >> n;
    for(int i = 0; i < n; i++)
        for(int j = 0; j < n; j++)
        g[i][j] = '.';
    dfs(0);
    return 0;
}

三、总结

在这里插入图片描述
    本文总结了几道bfs和dfs的经典例题, 比如红与黑这道题,我们用dfs和bfs两种方法去解决,算法是一个重要的学习模块,需要我们多刷题和总结,希望大家读后能够有所收获!!!

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

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

相关文章

2023最新SSM计算机毕业设计选题大全(附源码+LW)之java电子书店管理系统ya226

做毕业设计一定要选好题目。毕设想简单&#xff0c;其实很简单。这里给几点建议&#xff1a; 1&#xff1a;首先&#xff0c;学会收集整理&#xff0c;年年专业都一样&#xff0c;岁岁毕业人不同。很多人在做毕业设计的时候&#xff0c;都犯了一个错误&#xff0c;那就是不借鉴…

双功能接头试剂知识分享:Alkyne maleimide,Mal-Alkyne,炔烃-马来酰亚胺

炔烃马来酰亚胺是一种双功能接头试剂&#xff0c;可将末端炔烃连接到各种含硫醇分子&#xff0c;例如含有半胱氨酸残基的蛋白质。然后可以通过铜催化的点击化学反应将炔烃部分与各种叠氮化物缀合。 Alkyne maleimide is a bifunctional connector reagent that can connect ter…

AB Test实验设计

1. 版本设计 实验版本的设计要遵循变量的单一性&#xff0c;不能一下子改变多个因素&#xff0c;如同一个按钮不能同时改变按钮颜色和按钮文字&#xff0c;实验设计越简单越容易得出正确的结论。 案例时间&#xff1a; 2. 实验时长 业界的实验时长一般是2-3周&#xff0c;最…

Packet Tracer - 在 OSPFv2 中传播默认路由

地址分配表 设备 接口 IPv4 地址 子网掩码 默认网关 R1 G0/0 172.16.1.1 255.255.255.0 不适用 S0/0/0 172.16.3.1 255.255.255.252 不适用 S0/0/1 192.168.10.5 255.255.255.252 不适用 R2 G0/0 172.16.2.1 255.255.255.0 不适用 S0/0/0 172.16.3.2 …

Redis对象及redisObject源码解析

写在前面 以下内容是基于Redis 6.2.6 版本整理总结 一、对象 前面几篇文章&#xff0c;我们介绍了Redis用到的主要的数据结构&#xff0c;如&#xff1a;sds、list、dict、ziplist、skiplist、inset等。 但是&#xff0c;Redis并没有直接使用这些数据结构来实现key-value数…

使用k8s创建一个支持ssh的pod,docker

在容器场景下 用ssh去登录一个docker 是不提倡的 可是有时为了方便定位问题 我制作了一个支持ssh的镜像 节约大家的时间 docker使用方法: docker run --namec7-sshd --privileged -itd -p 1022:22 lnwaycool/centos7_ssh:v1 /usr/sbin/init ssh root127.0.0.1 -p 1022 登…

猴子也能学会的jQuery第七期——jQuery动画(上)

&#x1f4da;系列文章—目录&#x1f525; 猴子也能学会的jQuery第一期——什么是jQuery 猴子也能学会的jQuery第二期——引用jQuery 猴子也能学会的jQuery第三期——使用jQuery 猴子也能学会的jQuery第四期——jQuery选择器大全 猴子也能学会的jQuery第五期——jQuery样式操作…

外部H5页面打开微信小程序最新流程

当前时间2022年11月22日&#xff0c;记录一下外部H5如何打开指定微信小程序的指定页面最新流程。 因为微信小程序后台已关闭生成 小程序 scheme 码 入口&#xff0c;所以只能通过如下方式&#xff1a; 1.服务端获取微信小程序 scheme 码 前提条件&#xff1a;目前仅针对国内…

基于中国新能源汽车税收政策下成都市场发展路线研究

目 录 摘 要 I Abstract II 第一章 绪论 1 1.1选题背景及意义 1 1.2国内外新能源汽车税收政策现状 2 1.2.1国内新能源汽车税收政策 2 1.2.2国外新能源汽车税收政策 3 1.3本课题研究主要内容 4 第二章 新能源汽车市场发展现状及存在的问题 5 2.1市场发展现状 5 2.2存在的问题 5 …

TensorFlow之文本分类算法-2

1 前言 2 收集数据 3 探索数据 4 选择模型 5 准备数据 数据被输入模型之前&#xff0c;需要将数据转换成模型能理解的格式&#xff0c;该过程被称之为数据标准化。 首先&#xff0c;已收集到的数据样本可能是以指定的顺序存储&#xff0c;而在实际分析中&#xff0c;期望…

邮件助手、监控邮件上报电脑截图、网课监控助手

起因:看到别的网友写的一个程序,监控在家上网课的状态,防止孩子玩游戏。没界面无法配置更多参数。自己写了一个。功能基本齐全了。 收件地址:接收邮件的邮箱 抄送地址:可以多个邮箱,用|分开即可。 标题、内容、版权位置可以自定义内容。 设置发送频率(默认10分钟),…

深入浅出日志体系(logback最佳实践)

最近和大家说了不少“大问题”&#xff08;中庸之道啥的&#xff09;&#xff0c;今天想和大家聊一个“小问题”——如何写日志。 在很多人的认知里面&#xff0c;日志的确是可有可无的小问题&#xff0c;因为有没有日志都不影响业务功能的运行。正因为如此&#xff0c;日志问…

内存取证系列6

文章目录文档说明挑战说明解题过程查看系统镜像查看系统进程查看使用命令查看命令输出查看历史命令记录搜索压缩包 获取虚拟地址并导出本地扫描ie历史记录扫描firefox历史记录(默认不支持)导出浏览器数据分析浏览器数据解压压缩包 得到flag2继续分析浏览器数据修复图片 获得fla…

Oracle AutoVue 21.0.x最新支持程序文件格式及版本

目录 前言 Engineering & Construction / Energy / Utilities Industrial Manufacturing / Automotive / Aerospace & Defense Electronics & High Tech Desktop / Office Graphics Others 前言 AutoVue官是一个多用途、多功能的图…

linux系统中cpu性能优化

cpu性能: uptime命令中的系统平均负载: 平均负载时指的单位时间内,系统处于可运行状态和不可中断状态的平均进程数,也就是平均活跃进程数,它和CPU使用率并没有直接关系. 可运行状态的进程: 是指正在使用Cpu或者正在等待Cpu的进程,也就是我们ps命令看到的处于Runing状态的进程.…

物联网安全年报事件回顾

执行摘要 随着物联网的不断发展&#xff0c;物联网安全也被越来越多的人所关注。我们于 2016 年发布《物联网安全 白皮书》&#xff0c;进行物联网安全的科普介绍&#xff1b;于 2017 年发布《2017 物联网安全年报》&#xff0c;关注物联网资产 在互联网上的暴露情况、物联网设…

【CNN】GoogLeNet——大胆创新网络结构

前言 2014年&#xff0c;GoogLeNet和VGG是当年ImageNet挑战赛(ILSVRC14)的双雄&#xff0c;GoogLeNet获得了图片分类大赛第一名、VGG紧随其后&#xff0c;这两类模型结构的共同特点是网络深度更深了。VGG继承了LeNet以及AlexNet的一些框架结构&#xff0c;而GoogLeNet则做了更…

uniapp集成Android原生sdk

最近公司有个项目&#xff0c;需要调用客户提供的sdk扫描rfid&#xff0c;项目又是用uni开发的&#xff0c;客户提供的sdk只有Java版本&#xff0c;我发现uni是提供集成原生sdk的&#xff0c;所以研究了一下怎么使用&#xff0c;并记录下来过程。 准备 SDK 文件 下载 首先下…

VUE搭建后台管理界面

后台管理一、前言二、依赖配置三、koa框架四、数据库五、路由六、前端界面6.1 登录界面6.2 注册界面6.3 展示界面七、阿里云部署7.1 前端项目7.2 后端node7.3 mysql安装7.4 测试八、总结一、前言 本篇文章从头到尾过一遍vue3搭建项目的过程&#xff0c;实现一个用户登录、注册…

Linux上使用telnet连接本机IP地址端口

场景 Linux开启了iptables&#xff01;&#xff01;&#xff01; 开启本机TCP80端口服务。Linux本机IP地址是192.168.204.129。本机telnet连接本机的TCP80端口。 目的 telnet 192.168.204.129 80&#xff0c;能够连接通。 输入规则 1&#xff09;需要配置一个让本机TCP80端…