【Algorithm】广度优先搜索BFS

前面介绍了深度优先搜索,可知 DFS 是以深度作为第一关键词的,即当碰到岔道口时总是先选择其中的一条岔路前进,而不管其他岔路,直到碰到死胡同时才返回岔道口并选择其他岔道口。

接下来介绍的广度优先搜索则是以广度为第一关键词,当碰到岔路口时,总是先依次访问从该岔路口能直接到达的所有结点,然后再按这些结点被访问的顺序去依次访问它们能直接到达的所有结点,以此类推,直到所有的结点都被访问为止。这就跟平静的水面中投入一颗小石子一样,水花总是以石子落水处为中心,并以同心圆的方式向外扩散至整个水面,从这点看和 DFS 那种沿着一条线前进的思路是完全不同的。

 

#include <cstdio>
#include <queue>
using namespace std;

const int maxn = 100;
struct node {
    int x, y;
} Node;

int n, m;   // 矩阵大小 n * m
int matrix[maxn][maxn]; // 01 矩阵
bool inq[maxn][maxn] = { false };   // 记录位置 (x, y) 是否已入过队
int X[4] = {0, 0, 1, -1};   // 增量数组
int Y[4] = {1, -1, 0, 0};

bool judge(int x, int y)    // 判断坐标 (x, y) 是否需要访问
{
    // 越界返回 false
    if(x >= n || x < 0 || y >= m || y < 0) return false;
    // 当前位置为 0, 或 (x, y) 已入过队, 返回 false
    if(matrix[x][y] == 0 || inq[x][y] == true) return false;
    // 以上都不满足, 返回 true
    return true;
}

// BFS 函数访问位置 (x, y) 所在的块, 将该块中所有的 "1" 的 inq 都设置为 true
void BFS(int x, int y)
{
    queue<node> Q;
    Node.x = x, Node.y = y;
    Q.push(Node);
    inq[x][y] = true;
    while(!Q.empty()) {
        node top = Q.front();
        Q.pop();
        for(int i = 0; i < 4; i++) {
            int newX = top.x + X[i];
            int newY = top.y + Y[i];
            if(judge(newX, newY)) { // 如果新位置 (newX, newY) 需要访问
                Node.x = newX, Node.y = newY;
                Q.push(Node);
                inq[newX][newY] = true;
            }
        }
    }
}

int main()
{
    scanf("%d %d", &n, &m);
    for(int x = 0; x < n; x++) {
        for(int y = 0; y < m; y++) {
            scanf("%d", &matrix[x][y]);
        }
    }

    int ans = 0;    // 存入块数
    for(int x = 0; x < n; x++) {    // 枚举每一个位置
        for(int y = 0; y < m; y++) {
            // 如果元素为 1, 且未入过队
            if(matrix[x][y] == 1 && inq[x][y] == false) {
                ans++;  // 块数加 1
                BFS(x, y);  // 访问整个块, 将该块所有 "1" 的 inq 都标记为 true
            }
        }
    }

    printf("%d\n", ans);

    return 0;
}
/*
7 6
0 1 1 1 0 0 1
0 0 1 0 0 0 0
0 0 0 0 1 0 0
0 0 0 1 1 1 0
1 1 1 0 1 0 0
1 1 1 1 0 0 0 
 */

 

#include <cstdio>
#include <cstring>
#include <queue>
using namespace std;

const int maxn = 100;
struct node {
    int x, y;   // 位置 (x, y)
    int step;   // step 为从起点 S 到达该位置的最少步数 (层数)
} S, T, Node;   // S 为起点, T 为终点, Node 为临时结点

int n, m;   // n 行 m 列
char maze[maxn][maxn];  // 迷宫信息
bool inq[maxn][maxn] = { false };   // 记录位置 (x, y) 是否已入过队
int X[4] = {0, 0, 1, -1};
int Y[4] = {1, -1, 0, 0};

// 检查位置 (x, y) 是否有效
bool test(int x, int y)
{
    if(x >= n || x < 0 || y >= m || y < 0) return false;    // 超过边界
    if(maze[x][y] == '*') return false; //墙壁 *
    if(inq[x][y] == true) return false; // 已入过对
    return true;
}

int BFS()
{
    queue<node> q;
    q.push(S);
    while(!q.empty()) {
        node top = q.front();
        q.pop();
        if(top.x == T.x && top.y == T.y) {
            return top.step;    // 终点, 直接返回最少步数
        }
        for(int i = 0; i < 4; i++) {    // 循坏 4 次, 得到 4 个相邻位置
            int newX = top.x + X[i];
            int newY = top.y + Y[i];
            if(test(newX, newY)) {  // 位置 (newX, newY) 有效
                Node.x = newX, Node.y = newY;
                Node.step = top.step + 1;
                q.push(Node);
                inq[newX][newY] = true;
            }
        }
    }
    return -1;  // 无法到达最终 T 时返回 -1
}

int main()
{
    scanf("%d%d", &n, &m);
    for(int i = 0; i < n; i++) {
        getchar();  // 过滤掉每行后面的换行符
        for(int j = 0; j < m; j++) {
            maze[i][j] = getchar();
        }
        // maze[i][m + 1] = '\0';
    }
    scanf("%d%d%d%d", &S.x, &S.y, &T.x, &T.y);  // 起点和终点的坐标
    S.step = 0; // 起始化起点的层数为 0, 即 S 到 S 的最少步数为 0
    printf("%d\n", BFS());
    return 0;
}
/*
5 5
.....
.*.*.
.*S*.
...T*
2 2 4 3
*/

 

原文地址:https://blog.csdn.net/y__a__o/article/details/123583512

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


这篇文章主要介绍“基于nodejs的ssh2怎么实现自动化部署”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“基于nodejs...
本文小编为大家详细介绍“nodejs怎么实现目录不存在自动创建”,内容详细,步骤清晰,细节处理妥当,希望这篇“nodejs怎么实现目录不存在自动创建”文章能帮助大...
这篇“如何把nodejs数据传到前端”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这...
本文小编为大家详细介绍“nodejs如何实现定时删除文件”,内容详细,步骤清晰,细节处理妥当,希望这篇“nodejs如何实现定时删除文件”文章能帮助大家解决疑惑...
这篇文章主要讲解了“nodejs安装模块卡住不动怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来...
今天小编给大家分享一下如何检测nodejs有没有安装成功的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文...
本篇内容主要讲解“怎么安装Node.js的旧版本”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎...
这篇“node中的Express框架怎么安装使用”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家...
这篇文章主要介绍“nodejs如何实现搜索引擎”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“nodejs如何实现搜索引擎...
这篇文章主要介绍“nodejs中间层如何设置”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“nodejs中间层如何设置”文...
这篇文章主要介绍“nodejs多线程怎么实现”,在日常操作中,相信很多人在nodejs多线程怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法...
这篇文章主要讲解了“nodejs怎么分布式”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“nodejs怎么分布式”...
本篇内容介绍了“nodejs字符串怎么转换为数组”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情...
这篇文章主要介绍了nodejs如何运行在php服务器的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇nodejs如何运行在php服务器文章都...
本篇内容主要讲解“nodejs单线程如何处理事件”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“nodejs单线程如何...
这篇文章主要介绍“nodejs怎么安装ws模块”,在日常操作中,相信很多人在nodejs怎么安装ws模块问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法...
本篇内容介绍了“怎么打包nodejs代码”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!
本文小编为大家详细介绍“nodejs接收到的汉字乱码怎么解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“nodejs接收到的汉字乱码怎么解决”文章能帮助大家解...
这篇“nodejs怎么同步删除文件”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇...
今天小编给大家分享一下nodejs怎么设置淘宝镜像的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希