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

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

一、宽度遍历搜索(bfs)


1.概念和算法思想


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


2.红与黑(牛客网)


2.1题目链接红与黑


2.2分析题面

1669445671607.jpg

1669445699117.jpg

题意:输入对应的两个数m,和n,m是行, n是列,输入一个char类型的图,遍历这个图,找到‘@‘这个点,并且从’@‘点开始计算能走到的’.'字符的总个数。

算法分析:flood fill (大水漫灌算法)

bfs算法适合做最短路的问题,遍历顺序,从’@'这个点往上右下左四个方向去搜索,所以我们可以设置两个方向数组,就可以实现。bfs的实现还要依靠队列来完成,一般是默认是STL库里提供的队列就可以了。

1669445726258.jpg

这道题的实现思路:

1669445763853.jpg

代码示例:

#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分析题面

1669445818542.jpg

题意:入口固定在第一行第二列,出口固定在最后一行第九列,其中’#‘代表墙, '.'代表通路,算从入口到出口的最少步数u,就相当于是最短路,也能用bfs解。

算法分析:flood fill (大水漫灌算法)

解法与上题基本没区别,但是要注意bfs函数的实现。

1669445830954.jpg

代码示例:

#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分析题面

1669445862079.jpg

给定一个整数n,给出1 - n的全排列。

算法分析: dfs算法,顺序很重要

1669445874795.jpg

代码示例:

#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分析题面

1669445900031.jpg

1669445928369.jpg

题意: 输入对应的两个数m,和n,m是行, n是列,输入一个char类型的图,遍历这个图,找到‘@‘这个点,并且从’@‘点开始计算能走到的’.'字符的总个数。


3.2算法分析

1669445918034.jpg


代码示例:

#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分析题面

1669445969116.jpg

1669445981819.jpg

n个皇后放在n x n的国际象棋上,就是每一行要放一个皇后,但是任意一个皇后不能处在同一行和同一列和同一斜线上。


4.2算法分析(dfs)


使用类似全排列的方式,把符合条件的选出来。

1669445990960.jpg


代码示例:

#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;
}


相关文章
|
4天前
|
算法
分享一些提高二叉树遍历算法效率的代码示例
这只是简单的示例代码,实际应用中可能还需要根据具体需求进行更多的优化和处理。你可以根据自己的需求对代码进行修改和扩展。
|
7天前
|
存储 缓存 算法
如何提高二叉树遍历算法的效率?
选择合适的遍历算法,如按层次遍历树时使用广度优先搜索(BFS),中序遍历二叉搜索树以获得有序序列。优化数据结构,如使用线索二叉树减少空指针判断,自定义节点类增加辅助信息。利用递归与非递归的特点,避免栈溢出问题。多线程并行遍历提高速度,注意线程安全。缓存中间结果,避免重复计算。预先计算并存储信息,提高遍历效率。综合运用这些方法,提高二叉树遍历算法的效率。
23 5
|
7天前
|
算法
树的遍历算法有哪些?
不同的遍历算法适用于不同的应用场景。深度优先搜索常用于搜索、路径查找等问题;广度优先搜索则在图的最短路径、层次相关的问题中较为常用;而二叉搜索树的遍历在数据排序、查找等方面有重要应用。
14 2
|
10天前
|
机器学习/深度学习 JSON 算法
二叉树遍历算法的应用场景有哪些?
【10月更文挑战第29天】二叉树遍历算法作为一种基础而重要的算法,在许多领域都有着不可或缺的应用,它为解决各种复杂的问题提供了有效的手段和思路。随着计算机科学的不断发展,二叉树遍历算法也在不断地被优化和扩展,以适应新的应用场景和需求。
20 0
|
10天前
|
算法 搜索推荐 数据库
二分搜索:高效的查找算法
【10月更文挑战第29天】通过对二分搜索的深入研究和应用,我们可以不断挖掘其潜力,为各种复杂问题提供高效的解决方案。相信在未来的科技发展中,二分搜索将继续发挥着重要的作用,为我们的生活和工作带来更多的便利和创新。
16 1
|
1月前
|
算法 决策智能
基于禁忌搜索算法的VRP问题求解matlab仿真,带GUI界面,可设置参数
该程序基于禁忌搜索算法求解车辆路径问题(VRP),使用MATLAB2022a版本实现,并带有GUI界面。用户可通过界面设置参数并查看结果。禁忌搜索算法通过迭代改进当前解,并利用记忆机制避免陷入局部最优。程序包含初始化、定义邻域结构、设置禁忌列表等步骤,最终输出最优路径和相关数据图表。
|
1月前
|
存储 算法
数据结构与算法学习十六:树的知识、二叉树、二叉树的遍历(前序、中序、后序、层次)、二叉树的查找(前序、中序、后序、层次)、二叉树的删除
这篇文章主要介绍了树和二叉树的基础知识,包括树的存储方式、二叉树的定义、遍历方法(前序、中序、后序、层次遍历),以及二叉树的查找和删除操作。
23 0
|
22天前
|
算法 安全 数据安全/隐私保护
基于game-based算法的动态频谱访问matlab仿真
本算法展示了在认知无线电网络中,通过游戏理论优化动态频谱访问,提高频谱利用率和物理层安全性。程序运行效果包括负载因子、传输功率、信噪比对用户效用和保密率的影响分析。软件版本:Matlab 2022a。完整代码包含详细中文注释和操作视频。
|
7天前
|
算法 数据挖掘 数据安全/隐私保护
基于FCM模糊聚类算法的图像分割matlab仿真
本项目展示了基于模糊C均值(FCM)算法的图像分割技术。算法运行效果良好,无水印。使用MATLAB 2022a开发,提供完整代码及中文注释,附带操作步骤视频。FCM算法通过隶属度矩阵和聚类中心矩阵实现图像分割,适用于灰度和彩色图像,广泛应用于医学影像、遥感图像等领域。
|
8天前
|
算法 调度
基于遗传模拟退火混合优化算法的车间作业最优调度matlab仿真,输出甘特图
车间作业调度问题(JSSP)通过遗传算法(GA)和模拟退火算法(SA)优化多个作业在并行工作中心上的加工顺序和时间,以最小化总完成时间和机器闲置时间。MATLAB2022a版本运行测试,展示了有效性和可行性。核心程序采用作业列表表示法,结合遗传操作和模拟退火过程,提高算法性能。