c++算法学习4——广度搜索bfs

一、引言:探索迷宫的智能方法

在解决迷宫最短路径问题时,广度优先搜索(BFS)是一种高效而优雅的算法。与深度优先搜索(DFS)不同,BFS采用"由近及远"的搜索策略,逐层探索所有可能的路径,从而保证首次到达终点时的路径就是最短路径。(对深搜没有了解的同学,可以先看下我写的关于深搜的学习文章)

二、问题描述:迷宫寻路

给定一个R行C列的迷宫,迷宫由'.'(空地)和'#'(障碍物)组成。我们需要计算从左上角(入口)到右下角(出口)的最短步数(包括起点和终点)。移动规则:每次只能向上、下、左、右四个方向移动到相邻的空地格子。

输入示例

5 5
..###
#....
#.#.#
#.#.#
#.#..

输出示例

9

三、BFS算法核心思想

BFS采用队列(Queue) 这一数据结构实现"先进先出"的访问策略:

  1. 从起点开始,将其加入队列

  2. 取出队首元素,探索其所有相邻位置

  3. 将未访问的有效位置加入队列

  4. 重复直到找到终点或队列为空

这种策略确保算法优先访问距离起点更近的位置,因此首次到达终点时的路径必然是最短路径。

四、完整代码实现

#include <iostream>
#include <queue>
#include <cstring>
using namespace std;int R, C; // 迷宫的行数和列数
char maze[40][40]; // 存储迷宫
bool visited[40][40]; // 标记访问状态// 方向数组:右(0,1), 下(1,0), 上(0,-1), 左(-1,0)
int dx[] = {0, 1, 0, -1};
int dy[] = {1, 0, -1, 0};// 位置结构体:存储坐标和步数
struct Position {int x, y; // 当前坐标int steps; // 从起点到当前位置的步数
};int bfs() {queue<Position> q;// 起点入队:位置(0,0),步数为1(包括起点)q.push({0, 0, 1});visited[0][0] = true; // 标记起点已访问while (!q.empty()) {Position current = q.front();q.pop();// 到达终点:右下角(R-1, C-1)if (current.x == R - 1 && current.y == C - 1) {return current.steps;}// 探索四个方向for (int i = 0; i < 4; i++) {int nx = current.x + dx[i]; // 新位置的行坐标int ny = current.y + dy[i]; // 新位置的列坐标// 检查新位置是否有效if (nx < 0 || nx >= R || ny < 0 || ny >= C) continue; // 超出边界if (maze[nx][ny] == '#') continue; // 遇到障碍物if (visited[nx][ny]) continue; // 已访问过// 有效位置:标记并加入队列visited[nx][ny] = true;q.push({nx, ny, current.steps + 1});}}// 如果没有找到路径(题目保证有解,此返回值不会执行)return -1;
}int main() {// 读入迷宫大小cin >> R >> C;// 读入迷宫数据for (int i = 0; i < R; i++) {for (int j = 0; j < C; j++) {cin >> maze[i][j];}}// 初始化访问标记数组memset(visited, false, sizeof(visited));// 执行BFS并输出结果cout << bfs() << endl;return 0;
}

五、关键代码解析

1. 方向数组:简洁的方向控制

int dx[] = {0, 1, 0, -1};
int dy[] = {1, 0, -1, 0};

这两个数组定义了四个基本移动方向:

  • 右:(x+0, y+1)

  • 下:(x+1, y+0)

  • 左:(x+0, y-1)

  • 上:(x-1, y+0)

使用方向数组避免了重复的方向判断代码,使程序更简洁。

2. 位置结构体:信息封装

struct Position {int x, y; // 当前位置坐标int steps; // 从起点到当前位置的步数
};

结构体封装了位置信息和步数,便于在队列中存储和传递状态。

3. BFS核心逻辑:逐层探索

while (!q.empty()) {Position current = q.front();q.pop();// 检查是否到达终点// 探索四个方向for (int i = 0; i < 4; i++) {// 计算新位置// 检查新位置有效性(边界、障碍、访问状态)// 有效位置入队}
}

这是BFS的核心循环,每次从队列中取出最早加入的位置(距离起点最近),探索其所有相邻位置。

4. 访问标记:避免重复访问

visited[nx][ny] = true;

每个位置在首次访问时被标记,确保每个位置只被访问一次,避免无限循环和不必要的重复计算。

六、BFS与DFS的对比

特性BFSDFS
数据结构队列(Queue)栈(Stack)
搜索策略广度优先,逐层扩展深度优先,沿路径到底
空间复杂度O(最宽层的节点数)O(最大深度)
最短路径首次到达即是最短路径需要遍历所有路径找最短
适用场景无权图最短路径问题所有路径遍历,连通性检查

在迷宫最短路径问题中,BFS具有明显优势,因为它无需遍历所有路径就能找到最短路径。

七、总结

广度优先搜索是解决迷宫最短路径问题的经典算法,其核心在于:

  • 使用队列管理待访问位置

  • 逐层探索保证首次到达终点即是最短路径

  • 通过访问标记避免重复计算

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

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

相关文章

4.RV1126-OPENCV 图像轮廓识别

一.图像识别API 1.图像识别作用 它常用于视觉任务、目标检测、图像分割等等。在 OPENCV 中通常使用 Canny 函数、findContours 函数、drawContours 函数结合在一起去做轮廓的形检测。 2.常用的API findContours 函数&#xff1a;用于寻找图片的轮廓&#xff0c;并把所有的数…

Qt多线程访问同一个数据库源码分享(基于Sqlite实现)

Qt多线程访问同一个数据库源码分享&#xff08;基于Sqlite实现&#xff09; 一、实现难点线程安全问题死锁风险连接管理问题数据一致性性能瓶颈跨线程信号槽最佳实践建议 二、源码分享三、测试1、新建一个多线程类2、开启多线程插入数据 一、实现难点 多线程环境下多个线程同时…

双空间知识蒸馏用于大语言模型

Dual-Space Knowledge Distillation for Large Language Models 发表&#xff1a;EMNLP 2024 机构&#xff1a;Beijing Key Lab of Traffic Data Analysis and Mining 连接&#xff1a;https://aclanthology.org/2024.emnlp-main.1010.pdf 代码&#xff1a;GitHub - songmz…

贪心算法应用:多重背包启发式问题详解

贪心算法应用&#xff1a;多重背包启发式问题详解 多重背包问题是经典的组合优化问题&#xff0c;也是贪心算法的重要应用场景。本文将全面深入地探讨Java中如何利用贪心算法解决多重背包问题。 多重背包问题定义 **多重背包问题(Multiple Knapsack Problem)**是背包问题的变…

ES6 Promise 状态机

状态机&#xff1a;抽象的计算模型&#xff0c;根据特定的条件或者信号切换不同的状态 一、Promise 是什么&#xff1f; 简单来说&#xff0c;Promise 就是一个“承诺对象”。在ES6 里&#xff0c;有些代码执行起来需要点时间&#xff0c;比如加载文件、等待网络请求或者设置…

【Docker管理工具】部署Docker可视化管理面板Dpanel

【Docker管理工具】部署Docker可视化管理面板Dpanel 一、Dpanel介绍1.1 DPanel 简介1.2 主要特点 二、本次实践规划2.1 本地环境规划2.2 本次实践介绍 三、本地环境检查3.1 检查Docker服务状态3.2 检查Docker版本3.3 检查docker compose 版本 四、下载Dpanel镜像五、部署Dpanel…

最新研究揭示云端大语言模型防护机制的成效与缺陷

一项全面新研究揭露了主流云端大语言模型&#xff08;LLM&#xff09;平台安全机制存在重大漏洞与不一致性&#xff0c;对当前人工智能安全基础设施现状敲响警钟。该研究评估了三大领先生成式AI平台的内容过滤和提示注入防御效果&#xff0c;揭示了安全措施在阻止有害内容生成与…

docker中,容器时间和宿机主机时间不一致问题

win11下的docker中有个mysql。今天发现插入数据的时间不正确。后来发现原来是docker容器中的时间不正确。于是尝试了各种修改&#xff0c;什么run -e TZ"${tzutil /g}"&#xff0c;TZ"Asia/Shanghai"&#xff0c;还有初始化时带--mysqld一类的&#xff0c;…

uniapp实现的简约美观的星级评分组件

采用 uniapp 实现的一款简约美观的星级评分模板&#xff0c;提供丝滑动画效果&#xff0c;用户可根据自身需求进行自定义修改、扩展&#xff0c;纯CSS、HTML实现&#xff0c;支持web、H5、微信小程序&#xff08;其他小程序请自行测试&#xff09; 可到插件市场下载尝试&#x…

go语言的锁

本篇文章主要讲锁&#xff0c;主要会涉及go的sync.Mutex和sync.RWMutex。 一.锁的概念和发展 1.1 锁的概念 所谓的加锁和解锁其实就是指一个数据是否被占用了&#xff0c;通过Mutex内的一个状态来表示。 例如&#xff0c;取 0 表示未加锁&#xff0c;1 表示已加锁&#xff…

Ubuntu 服务器软件更新,以及常用软件安装 —— 一步一步配置 Ubuntu Server 的 NodeJS 服务器详细实录 3

前言 前面&#xff0c;我们已经 安装好了 Ubuntu 服务器系统&#xff0c;并且 配置好了 ssh 免密登录服务器 &#xff0c;现在&#xff0c;我们要来进一步的设置服务器。 那么&#xff0c;本文&#xff0c;就是进行服务器的系统更新&#xff0c;以及常用软件的安装 调整 Ubu…

如何从零开始建设一个网站?

当你没有建站的基础和建站的知识&#xff0c;那么应该如何开展网站建设和网站管理。而今天的教程是不管你是为自己建站还是为他人建站都适合的。本教程会指导你如何进入建站&#xff0c;将建站的步骤给大家分解&#xff1a; 首先我们了解一下&#xff0c;建站需要那些步骤和流程…

网络可靠性的定义与核心要素

网络可靠性&#xff08;Network Reliability&#xff09;是指网络系统在特定时间范围内持续提供稳定、无中断、符合预期性能的服务能力。其核心目标是确保数据能够准确、完整、及时地传输&#xff0c;即使在部分故障或异常情况下仍能维持基本功能。 1. 网络可靠性的核心指标 衡…

GpuGeek如何成为AI基础设施市场的中坚力量

AI时代&#xff0c;算力基础设施已成为支撑技术创新和产业升级的关键要素。作为国内专注服务算法工程师群体的智算平台&#xff0c;GpuGeek通过持续创新的服务模式、精准的市场定位和系统化的生态建设&#xff0c;正快速成长为AI基础设施领域的中坚力量。本文将深入分析GpuGeek…

【Qt】Bug:findChildren找不到控件

使用正确的父对象调用 findChildren&#xff1a;不要在布局对象上调用 findChildren&#xff0c;而应该在布局所在的窗口或控件上调用。

【Linux网络编程】传输层协议TCP,UDP

目录 一&#xff0c;UDP协议 1&#xff0c;UDP协议的格式 2&#xff0c;UDP的特点 3&#xff0c;面向数据报 4&#xff0c;UDP的缓冲区 5&#xff0c;UDP使用注意事项 6&#xff0c;基于UDP的应用层协议 二&#xff0c;对于报文的理解 三&#xff0c;TCP协议 1&…

Neo4j 数据可视化与洞察获取:原理、技术与实践指南

在关系密集型数据的分析领域,Neo4j 凭借其强大的图数据模型脱颖而出。然而,将复杂的连接关系转化为直观见解,需要专业的数据可视化技术和分析方法。本文将深入探讨 Neo4j 数据可视化的核心原理、关键技术、实用技巧以及结合图数据科学库(GDS)获取深度洞察的最佳实践。 Ne…

树莓派超全系列教程文档--(55)如何使用网络文件系统NFS

如何使用网络文件系统NFS 网络文件系统 (NFS)设置基本 NFS 服务器Portmap 锁定&#xff08;可选&#xff09; 配置 NFS 客户端端口映射锁定&#xff08;可选&#xff09; 配置复杂的 NFS 服务器组权限DNS&#xff08;可选&#xff0c;仅在使用 DNS 时&#xff09;NIS&#xff0…

无法运用pytorch环境、改环境路径、隔离环境

一.未建虚拟环境时 1.创建新项目后&#xff0c;直接运行是这样的。 2.设置中Virtualenv找不到pytorch环境&#xff1f;因为此时没有创建新虚拟环境。 3.选择conda环境&#xff08;全局环境&#xff09;时&#xff0c;是可以下载环境的。 运行结果如下&#xff1a; 是全局环境…

HTML5+CSS3+JS小实例:具有粘性重力的磨砂玻璃导航栏

实例:具有粘性重力的磨砂玻璃导航栏 技术栈:HTML+CSS+JS 效果: 源码: 【HTML】 <!DOCTYPE html> <html lang="zh-CN"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width…