【搜索与图论】BFS(广度优先搜索)

一、定义

从起始节点开始,首先访问所有与起始节点距离为 1 的邻居节点,然后依次访问距离为 2、距离为 3...... 的节点,直到遍历完整个图或找到目标节点为止。

二、核心原理

BFS 算法通常依赖**队列(Queue)**这一数据结构来维护待访问节点的顺序。

使用bfs算法处理的不管是图、树,还是迷宫问题,本质都是"队列+访问标记"的组合,

  1. 初始化:创建一个队列,将「起始节点」加入队列,并标记该节点为"已访问";
  2. 循环: 只要队列不为空,就执行以下操作:
    1. **出队:**取出队列头部的节点(当前节点);
    2. **处理:**对当前节点进行需要的操作(比如记录路径、判断是否是目标节点);
    3. **入队:**找到当前节点的所有"未访问"的邻居节点,标记为"已访问",并依次加入队列;
  3. **终止:**当队列为空,或者找到目标节点时,遍历结束。

三、例题

走迷宫

题目描述(来源于AcWing)

思路

  1. 用二维数组 g[N][N] 存迷宫地图(0 能走,1 是墙),dist[N][N] 存每个点到起点的距离,用队列 queue<PII> 存待遍历的坐标,用方向数组 dx[]、dy[] 控制上下左右移动。

  2. 初始化

    1. 把所有点的距离初始化为 -1(表示没走过)

    2. 起点 (1,1) 入队

    3. 起点距离设为 0(自己到自己步数为 0)

  3. bfs搜索**(只要队列不为空,就重复)**

    1. 取出队头坐标,作为当前位置

    2. 往上下左右 4 个方向尝试走一步

    3. 对下一步坐标做 3 个判断:

      • 没越界

      • 没走过(dist = -1)

      • 不是墙(g = 0)

    4. 满足条件就:

      • 记录距离 = 当前距离 + 1

      • 把新坐标加入队列

    5. 如果走到终点 (n,m),直接返回距离(BFS 第一次到达就是最短)

代码

cpp 复制代码
#include<iostream>
#include<algorithm>
#include<cstring>
#include<queue>
using namespace std;

typedef pair<int, int> PII;// 存储坐标
const int N = 110;
int n, m;
int g[N][N];//用于存储地图
int dist[N][N];//用于存储每个点到起点的距离
queue<PII> q; 
//定义方向数组
int dx[] = {-1,0,1,0};
int dy[] = {0,1,0,-1};

//两个参数是起点的坐标
int bfs(int x, int y){
    //初始化所有点到起点的距离为-1   ----这里使用memset函数
    memset(dist, -1, sizeof(dist));
    //起点入队列
    q.push({x,y});
    //修改距离数组
    dist[x][y] = 0;
    
    //当队列不为空的时候开始广度搜索
    while(!q.empty()){
        //取出队头
        auto t = q.front();
        //出队
        q.pop();
        
        //进行向上下左右四个方向走
        for(int i = 0; i < 4; i++){
            //求出下一步的横纵坐标
            int a = t.first + dx[i];
            int b = t.second + dy[i];
            
            //判断是否越界
            if(a < 1 || a > n || b < 1 || b > m){
                continue;
            }
            //判断该点是否已经走过
            if(dist[a][b] > -1){
                continue;
            }
            //判断g数组是否为0
            if(g[a][b] != 0){
                continue;
            }
            
            //将更新后的坐标进队列
            q.push({a,b});
            //更新距离数组
            dist[a][b] = dist[t.first][t.second] + 1;
            
            //如果已经到(n,m)了
            if(a == n && b == m){
                //返回结果
                return dist[n][m];
            }
        }
    }
    //返回距离
    return dist[n][m];
}
int main(){
    cin>>n>>m;
    
    //输入地图
    for(int i = 1; i <= n; i++){
        for(int j = 1; j <= m; j++){
            cin>>g[i][j];
        }
    }
    
    //广度优先搜索--- 得到最少移动次数
    int res = bfs(1,1);
    
    //输出结果
    cout<< res <<endl;
}

四、总结

如果每条边的权重都相等,则bfs算法可以求的最短路。

相关推荐
Proxy_ZZ011 分钟前
打造自己的信道编码工具箱——Turbo、LDPC、极化码三合一
c语言·算法·信息与通信
wayz1113 分钟前
21天机器学习核心算法学习计划(量化方向)
学习·算法·机器学习
穿条秋裤到处跑15 分钟前
每日一道leetcode(2026.04.09):区间乘法查询后的异或 II
算法·leetcode
超级大只老咪22 分钟前
一维度前缀和解题通用模板(java)
java·开发语言·算法
weixin_5134499626 分钟前
walk_these_ways项目学习记录第十篇(通过行为多样性 (MoB) 实现地形泛化)--从仿真到部署
人工智能·学习·算法
小欣加油31 分钟前
leetcode 42 接雨水
c++·算法·leetcode·职场和发展
tankeven37 分钟前
动态规划专题(14):石子合并问题(未完待续)
c++·算法·动态规划
像素猎人1 小时前
大学算法类竞赛的常用模板【自己总结+收录的】【c++版】
数据结构·算法·排序算法·算法竞赛常用算法
码云数智-大飞1 小时前
分布式锁的“双雄对决”:Redis 与 ZooKeeper 的深度博弈与选型指南
算法
cpp_25011 小时前
P1910 L 国的战斗之间谍
数据结构·c++·算法·题解·洛谷·背包dp