当前位置: 首页 > news >正文

BFS解决最短路问题(详解)

目录

BFS简介 && 框架:

一.二叉树的最小深度 

二:迷宫中里入口最近的出口:

 三.最小基因变化:

 四:单词接龙:

​五:为高尔夫比赛砍树:


BFS简介 && 框架:

说到BFS,想必大家都不陌生,我们在学习二叉树的遍历时所用到的层序遍历其实就是BFS,那么BFS能解决什么样的问题呢?通过本文标题,你也大致能猜到----解决最短路问题,这是一个比较经典的问题,当然,BFS还能解决很多其他问题,本文就着重介绍其中一种---最短路问题。

BFS基本概念:

  • BFS(广度优先搜索):是一种图形搜索算法,它从根节点开始,逐层地向下扩展搜索。BFS通常用队列来实现,即先进先出的数据结构。这意味着每个节点都将按照它们被发现的顺序进行处理。具体地说,BFS算法会首先访问根节点,然后将其所有相邻的节点加入队列中。接下来,它将按照与队列中节点相同的顺序访问这些新节点,并将它们的相邻节点加入队列中。该过程一直持续到所有节点都被访问为止。

我们先举例一下 BFS 出现的常见场景吧,问题的本质就是让你在一幅「图」中找到从起点 start 到终点 target 的最近距离,这个例子听起来很枯燥,但是 BFS 算法问题其实都是在干这个事儿 ,其对应的大致框架如下:

// 计算从起点 start 到终点 target 的最近距离
int BFS(Node start, Node target) {Queue<Node> q; // 核心数据结构Set<Node> visited; // 避免走回头路q.offer(start); // 将起点加入队列visited.add(start);while (q not empty) {int sz = q.size();/* 将当前队列中的所有节点向四周扩散 */for (int i = 0; i < sz; i++) {Node cur = q.poll();/* 划重点:这里判断是否到达终点 */if (cur is target)return step;/* 将 cur 的相邻节点加入队列 */for (Node x : cur.adj()) {if (x not in visited) {q.offer(x);visited.add(x);}}}}// 如果走到这里,说明在图中没有找到目标节点
}

一.二叉树的最小深度 

先来个简单的问题实践一下 BFS 框架吧,判断一棵二叉树的最小高度,这也是力扣第 111 题「二叉树的最小深度open in new window」: 

题目描述:

给定一个二叉树,找出其最小深度。

最小深度是从根节点到最近叶子节点的最短路径上的节点数量。

说明:叶子节点是指没有子节点的节点。

这题比较简单,直接套用上面的框架即可,遇到第一个叶子节点直接返回对应深度即可:

代码详解:

class Solution {public int minDepth(TreeNode root) {Queue<TreeNode> q = new LinkedList<>();if(root == null){return 0;}int depth = 1;//初始树的深度是1q.offer(root);while(!q.isEmpty()){int sz = q.size();for(int i = 0;i < sz;i++){TreeNode cur = q.poll();if(cur.left == null && cur.right == null){return depth;}//----->如果cur == null --->//  cur.left不存在,null指针异常/*       q.offer(cur.left);q.offer(cur.right);不能直接将节点放入队列,*/if(cur.left != null){//要判空q.offer(cur.left);}if(cur.right != null){q.offer(cur.right);}}depth++;}return depth;}
}

运行结果:

这里注意这个 while 循环和 for 循环的配合,while 循环控制一层一层往下走,for 循环利用 sz 变量控制从左到右遍历每一层二叉树节点,将节点加入到队列中

二:迷宫中里入口最近的出口:

题目链接:1926. 迷宫中离入口最近的出口 - 力扣(LeetCode)

题目描述:

给你一个 m x n 的迷宫矩阵 maze (下标从 0 开始),矩阵中有空格子(用 '.' 表示)和墙(用 '+' 表示)。同时给你迷宫的入口 entrance ,用 entrance = [entrancerow, entrancecol] 表示你一开始所在格子的行和列。

每一步操作,你可以往  或者  移动一个格子。你不能进入墙所在的格子,你也不能离开迷宫。你的目标是找到离 entrance 最近 的出口。出口 的含义是 maze 边界 上的 空格子entrance 格子 不算 出口。 

请你返回从 entrance 到最近出口的最短路径的 步数 ,如果不存在这样的路径,请你返回 -1 。

思路:我们可以从起点开始层序遍历,并且在遍历的过程中记录当前遍历的层数。这样就能在找到出⼝的 时候,得到起点到出⼝的最短距离

其中,为了方便表示四个方向,我们可以通过定义两个数组来表示方向:dx[ ],dy[ ],其中dx[ ],dy[ ]的位置要一一对应,具体操作如下:

代码详解:

class Solution {int[] dx = {0,0,-1,1};int[] dy = {1,-1,0,0};boolean[][] used;public int nearestExit(char[][] maze, int[] e) {int m = maze.length,n = maze[0].length;used = new boolean[m][n];//HashSet -- > Set来记录路径(走过的就不要再去走了)Queue<int[]> q = new LinkedList<>(); //int[]用来记录maze的下标q.offer(new int[]{e[0],e[1]});used[e[0]][e[1]] = true;int step = 0;//初始步数是0while(!q.isEmpty()){int sz = q.size();step++;for(int i = 0;i < sz;i++){int[] arr = q.poll();int a = arr[0],b = arr[1];for(int k = 0;k < 4;k++){//四个方向int x = dx[k] + a,y = dy[k] + b;if(x >= 0 && x < m && y >= 0 && y < n && maze[x][y] == '.' && !used[x][y]){if(x == 0 || x == m - 1 || y == 0 || y == n - 1){return step;}q.offer(new int[]{x,y});used[x][y] = true;}}}}return -1;//找不到就返回-1}
}

运行结果:

 三.最小基因变化:

题目链接:433. 最小基因变化 - 力扣(LeetCode)

题目描述:

基因序列可以表示为一条由 8 个字符组成的字符串,其中每个字符都是 'A''C''G' 和 'T' 之一。

假设我们需要调查从基因序列 start 变为 end 所发生的基因变化。一次基因变化就意味着这个基因序列中的一个字符发生了变化。

  • 例如,"AACCGGTT" --> "AACCGGTA" 就是一次基因变化。

另有一个基因库 bank 记录了所有有效的基因变化,只有基因库中的基因才是有效的基因序列。(变化后的基因必须位于基因库 bank 中)

给你两个基因序列 start 和 end ,以及一个基因库 bank ,请你找出并返回能够使 start 变化为 end 所需的最少变化次数。如果无法完成此基因变化,返回 -1 。

注意:起始基因序列 start 默认是有效的,但是它并不一定会出现在基因库中。

思路:用哈希表来记录搜索过的状态,存储基因库的信息(后续O(1)的时间查找),遍历一个字符串,将每个位置的可能的四种情况全部枚举出来,将没有被搜过或者存在基因库中的节点加入到队列。

代码详解:

class Solution {public int minMutation(String startGene, String endGene, String[] bank) {Set<String> hash = new HashSet<>();Set<String> vis = new HashSet<>();for(String x : bank) hash.add(x);char[] change = {'A','C','G','T'};if(!hash.contains(endGene)) return -1;if(endGene.equals(startGene)) return 0;Queue<String> q = new LinkedList<>();q.offer(startGene);vis.add(startGene);int step = 0;while(!q.isEmpty()){step++;int sz = q.size();while(sz-- != 0){String cur = q.poll();for(int i = 0;i < 8;i++){char[] temp = cur.toCharArray();for(int j = 0;j < 4;j++){temp[i] = change[j];String next = new String(temp);if(!vis.contains(next) && hash.contains(next)){if(next.equals(endGene)) return step;q.offer(next);vis.add(next);}}}}}return -1;}
}

运行结果:

 四:单词接龙:

题目链接:127. 单词接龙 - 力扣(LeetCode)

题目描述:

字典 wordList 中从单词 beginWord 和 endWord 的 转换序列 是一个按下述规格形成的序列 beginWord -> s1 -> s2 -> ... -> sk

  • 每一对相邻的单词只差一个字母。
  •  对于 1 <= i <= k 时,每个 si 都在 wordList 中。注意, beginWord 不需要在 wordList 中。
  • sk == endWord

给你两个单词 beginWord 和 endWord 和一个字典 wordList ,返回 从 beginWord 到 endWord 的 最短转换序列 中的 单词数目 。如果不存在这样的转换序列,返回 0 。

与上面一题基本类似,直接看代码:

class Solution {public int ladderLength(String beginWord, String endWord, List<String> wordList) {                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      Set<String> hash = new HashSet<>();//用于记录单词库中的单词Set<String> vis = new HashSet<>();//用于记录路径for(String x : wordList) hash.add(x);char[] change = {'a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','w','x','y','z'};//记录要该表的列表//处理一下边界条件//if(beginWord.equals(endWord)) return 1; -->题目给出不会相等if(!hash.contains(endWord)) return 0;//bfsQueue<String> q = new LinkedList<>();q.offer(beginWord);vis.add(beginWord);int step = 1;//用于记录单词数while(!q.isEmpty()){step++;int sz = q.size();while(sz-- != 0){String cur = q.poll();int k = cur.length();for(int i = 0;i < k;i++){char[] temp = cur.toCharArray();for(int j = 0;j < 26;j++){temp[i] = change[j];String next = new String(temp);//满足条件入队列if(!vis.contains(next) && hash.contains(next)){if(next.equals(endWord)) return step;//找到直接返回单词数q.offer(next);vis.add(next);}}}}}return 0;}
}

运行结果:

五:为高尔夫比赛砍树:

题目链接:675. 为高尔夫比赛砍树 - 力扣(LeetCode)

题目描述:

你被请来给一个要举办高尔夫比赛的树林砍树。树林由一个 m x n 的矩阵表示, 在这个矩阵中:

  • 0 表示障碍,无法触碰
  • 1 表示地面,可以行走
  • 比 1 大的数 表示有树的单元格,可以行走,数值表示树的高度

每一步,你都可以向上、下、左、右四个方向之一移动一个单位,如果你站的地方有一棵树,那么你可以决定是否要砍倒它。

你需要按照树的高度从低向高砍掉所有的树,每砍过一颗树,该单元格的值变为 1(即变为地面)。

你将从 (0, 0) 点开始工作,返回你砍完所有树需要走的最小步数。 如果你无法砍完所有的树,返回 -1 。

可以保证的是,没有两棵树的高度是相同的,并且你至少需要砍倒一棵树。

思路:

a. 先找出砍树的顺序;

b. 然后按照砍树的顺序,⼀个⼀个的⽤ bfs 求出最短路即可。 

代码详解:

class Solution {int m,n;public int cutOffTree(List<List<Integer>> forest) {m = forest.size();n = forest.get(0).size();//将是树的位置装入容器中List<int[]> trees = new ArrayList<>();for(int i = 0;i < m;i++){for(int j = 0;j < n;j++){if(forest.get(i).get(j) > 1){trees.add(new int[]{i,j});}}}//排序,按照树的相对高度对树对应坐标排序Collections.sort(trees,(a,b) ->{return forest.get(a[0]).get(a[1]) - forest.get(b[0]).get(b[1]);});int res = 0;int bx = 0,by = 0;for(int[] tree : trees){int x = tree[0],y = tree[1];//拿出下一个要去的位置的下表int step = bfs(forest,bx,by,x,y);//迷宫问题if(step == -1) return -1;res += step;//所有路径的最小值的和,就是答案bx = x;by = y;//更新起始位置的值}return res;}//迷宫问题->找到目标位置的最小步数int[] dx = {0,0,-1,1};int[] dy = {1,-1,0,0};public int bfs(List<List<Integer>> forest,int bx,int by,int ex,int ey){if(bx == ex && by == ey) return 0;//如果目标就在原地,直接返回boolean[][] used = new boolean[m][n];Queue<int[]> q = new LinkedList<>();q.offer(new int[]{bx,by});used[bx][by] = true;int step = 0;while(!q.isEmpty()){step++;int sz = q.size();while(sz-- != 0){int[] temp = q.poll();int a = temp[0],b = temp[1];for(int i = 0;i < 4;i++){int x = a + dx[i],y = b + dy[i];if(x >= 0 && x < m && y >= 0 && y < n && !used[x][y] && forest.get(x).get(y) != 0){if(x == ex && y == ey) return step;q.offer(new int[]{x,y});used[x][y] = true;}}}}return -1;//没找到就返回-1}
}

结语: 写博客不仅仅是为了分享学习经历,同时这也有利于我巩固知识点,总结该知识点,由于作者水平有限,对文章有任何问题的还请指出,接受大家的批评,让我改进。同时也希望读者们不吝啬你们的点赞+收藏+关注,你们的鼓励是我创作的最大动力!

相关文章:

BFS解决最短路问题(详解)

目录 BFS简介 && 框架&#xff1a; 一.二叉树的最小深度 二&#xff1a;迷宫中里入口最近的出口&#xff1a; 三.最小基因变化: 四&#xff1a;单词接龙&#xff1a; ​五&#xff1a;为高尔夫比赛砍树&#xff1a; BFS简介 && 框架&#xff1a; 说到BFS…...

按尺寸筛选轮廓图中的轮廓

1.按短边筛选 原始轮廓图&#xff1a; import cv2 import numpy as np# 读取轮廓图 contour_image cv2.imread(..\\IMGS\\pp_edge.png, cv2.IMREAD_GRAYSCALE)# 使用cv2.findContours()函数获取所有轮廓 contours, _ cv2.findContours(contour_image, cv2.RETR_EXTERNAL, cv2…...

VBA高级应用30例:实现在列表框内及列表框间实现数据拖动

《VBA高级应用30例》&#xff08;版权10178985&#xff09;&#xff0c;是我推出的第十套教程&#xff0c;教程是专门针对高级学员在学习VBA过程中提高路途上的案例展开&#xff0c;这套教程案例与理论结合&#xff0c;紧贴“实战”&#xff0c;并做“战术总结”&#xff0c;以…...

「AIGC算法」R-tree算法

R-tree算法是一种非常实用的空间数据索引技术,它可以帮助我们在复杂的空间数据中快速找到我们想要的信息。下面我将用一些生活中的例子来帮助大家更好地理解R-tree算法。 1. 定义与原理 想象一下,你有一个巨大的图书馆,里面有成千上万本书,每本书都有它在书架上的特定位置…...

2024软考上半年嵌入式系统设计师考试回顾

一&#xff1a;考试准备工作 1&#xff1a;基本上都是提前30分钟进考场&#xff0c;进入考试教室的时候&#xff0c;会有监考老师核对身份证和准考证&#xff1b; 2&#xff1a;进入考试教室之后&#xff0c;会再一次核对身份信息&#xff0c;并且有监考老师手持扫描仪&#x…...

MIT6.828 Lab2-1 Using gdb

Using gdb gdb使用&#xff1a; xv6 gdb调试方法 问题1&#xff1a; Looking at the backtrace output, which function called syscall? 按照提示开启gdb后键入&#xff1a; b syscall c layout src backtrace输出结果&#xff1a; (gdb) backtrace #0 syscall () at k…...

mysqldump提示Using a password on the command line interface can be insecured的解决办法

mysql数据库备份一句话执行命令 mysqldump --all-databases -h127.0.0.1 -uroot -p123456 > allbackupfile.sql 提示如下提示 [rootyfvyy5b2on3knb8q opt]# mysqldump --all-databases -h127.0.0.1 > allbackupfile.sql mysqldump: Couldnt execute SELECT COLUMN_NA…...

Java毕业设计 基于springboot vue考勤管理系统

Java毕业设计 基于springboot vue考勤管理系统 SpringBoot 考勤管理系统 功能介绍 员工 登录 个人中心 修改密码 个人信息 员工请假管理 员工出差管理 薪资管理 员工签到管理 公告管理 管理员 登录 个人中心 修改密码 个人信息 员工管理 员工请假管理 员工出差管理 薪资管理…...

C数据结构:二叉树

目录 二叉树的数据结构 前序遍历 中序遍历 后序遍历 二叉树的创建 二叉树的销毁 二叉树的节点个数 二叉树叶子节点个数 二叉树第K层节点个数 二叉树的查找 层序遍历 判断二叉树是否为完全二叉树 完整代码 二叉树的数据结构 typedef char BTDataType; typedef str…...

使用Nginx作为反向代理实现MQTT内外网通信

使用Nginx作为反向代理实现MQTT内外网通信 步骤1: 安装Nginx 确保你的服务器上已安装Nginx。如果未安装&#xff0c;可以通过以下命令在Ubuntu上安装Nginx&#xff1a; sudo apt update sudo apt install nginx步骤2: 配置Nginx 编辑Nginx的配置文件&#xff0c;通常是/etc…...

SpringBoot 上传文件示例

示例效果&#xff1a; 前端代码&#xff1a; <html> <head><title>上传文件示例</title></head> <body> <h2>方式一&#xff1a;普通表单上传</h2> <form action"/admin/upload" method"post" enctyp…...

9.js函数

函数是js复杂数据类型的一种---可以理解为存放代码的盒子 用来帮助我们封装、复用、扩展以及调用代码的工具 函数的两个阶段 &#xff08;1&#xff09;声明函数&#xff08;理解为创造&#xff09; ——声明式声明 语法&#xff1a;function 函数名(参数){...代码} ——赋值时…...

关于数据库和数据表的基础SQL

目录 一. 数据库的基础SQL 1. 创建数据库 2. 查看当前有哪些数据库 3. 选中数据库 4. 删除数据库 5. 小结 二. 数据表的基础SQL 1. 创建数据表 2. 查看当前数据库中有哪些表 3. 查看指定表的详细情况(查看表的结构) 4. 删除表 5. 小结 一. 数据库的基础SQL 1. 创建…...

【C语言深度解剖】(14):结构体内存对齐(详细配图讲解)

&#x1f921;博客主页&#xff1a;醉竺 &#x1f970;本文专栏&#xff1a;《C语言深度解剖》 &#x1f63b;欢迎关注&#xff1a;感谢大家的点赞评论关注&#xff0c;祝您学有所成&#xff01; ✨✨&#x1f49c;&#x1f49b;想要学习更多C语言深度解剖点击专栏链接查看&…...

学习笔记:C语言的32个关键字

一、标准C语言的32个关键字 1、基本数据类型&#xff1a; signed unsigned char int float double short long void 2、构造数据类型&#xff1a; struct union enum 3、数据存储类别&#xff1a; auto static extern register 4、数据优化&#xff1a; const volatile 5、9条…...

嵌入式学习 (Day:27 IPC --- 进程间通信)

IPC 进程间通信 interprocess communicate &#xff08;即&#xff1a;进程间进行数据交换&#xff09; 三大类&#xff1a; 进程间通信的方式&#xff08;共8种&#xff09; 1、古老的通信方式&#xff08;Linux设计时就有的&#xff09; 无名管道 有名…...

Python考试复习--day2

1.出租车计费 mile,waitmap(int,input().split(,)) if mile<3:money13wait*1 elif mile>3 and mile<15:money13(mile-3)*2.3wait*1 else:money1312*2.3(mile-15)*2.3*(10.5)wait*1 print({:.0f}.format(money)) 【知识点1】&#xff1a; map() 函数 【知识点1】&…...

整理好了!2024年最常见 20 道 Redis面试题(九)

上一篇地址&#xff1a;整理好了&#xff01;2024年最常见 20 道 Redis面试题&#xff08;八&#xff09;-CSDN博客 十七、Redis 的过期策略有哪些&#xff1f; Redis 的过期策略主要有三种&#xff1a; 定时删除&#xff1a;当为一个键设置了过期时间后&#xff0c;Redis 会…...

IDEA使用Maven打包项目的所有的依赖

要使用 Maven 命令将 Spring Boot 项目的依赖打包到 lib 文件夹中&#xff0c;你可以在终端中运行以下命令&#xff1a; mvn dependency:copy-dependencies -DoutputDirectory./lib这个命令会将项目的所有依赖&#xff08;包括运行时依赖&#xff09;复制到当前目录的 lib 文件…...

【C++ 】学习问题及补充

一.自定义类型不初始化直接就赋值&#xff0c;比如string类会怎么样 vectr<string>里已经给每个string对象已经分配好空间&#xff0c;为什么不初始化再赋值会报错 在C中&#xff0c;std::string类是一个动态字符串类&#xff0c;它内部管理着一个字符数组&#xff0c;用…...

[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?

&#x1f9e0; 智能合约中的数据是如何在区块链中保持一致的&#xff1f; 为什么所有区块链节点都能得出相同结果&#xff1f;合约调用这么复杂&#xff0c;状态真能保持一致吗&#xff1f;本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里&#xf…...

css实现圆环展示百分比,根据值动态展示所占比例

代码如下 <view class""><view class"circle-chart"><view v-if"!!num" class"pie-item" :style"{background: conic-gradient(var(--one-color) 0%,#E9E6F1 ${num}%),}"></view><view v-else …...

MySQL 隔离级别:脏读、幻读及不可重复读的原理与示例

一、MySQL 隔离级别 MySQL 提供了四种隔离级别,用于控制事务之间的并发访问以及数据的可见性,不同隔离级别对脏读、幻读、不可重复读这几种并发数据问题有着不同的处理方式,具体如下: 隔离级别脏读不可重复读幻读性能特点及锁机制读未提交(READ UNCOMMITTED)允许出现允许…...

以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:

一、属性动画概述NETX 作用&#xff1a;实现组件通用属性的渐变过渡效果&#xff0c;提升用户体验。支持属性&#xff1a;width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项&#xff1a; 布局类属性&#xff08;如宽高&#xff09;变化时&#…...

《基于Apache Flink的流处理》笔记

思维导图 1-3 章 4-7章 8-11 章 参考资料 源码&#xff1a; https://github.com/streaming-with-flink 博客 https://flink.apache.org/bloghttps://www.ververica.com/blog 聚会及会议 https://flink-forward.orghttps://www.meetup.com/topics/apache-flink https://n…...

HTML前端开发:JavaScript 常用事件详解

作为前端开发的核心&#xff0c;JavaScript 事件是用户与网页交互的基础。以下是常见事件的详细说明和用法示例&#xff1a; 1. onclick - 点击事件 当元素被单击时触发&#xff08;左键点击&#xff09; button.onclick function() {alert("按钮被点击了&#xff01;&…...

Springboot社区养老保险系统小程序

一、前言 随着我国经济迅速发展&#xff0c;人们对手机的需求越来越大&#xff0c;各种手机软件也都在被广泛应用&#xff0c;但是对于手机进行数据信息管理&#xff0c;对于手机的各种软件也是备受用户的喜爱&#xff0c;社区养老保险系统小程序被用户普遍使用&#xff0c;为方…...

AI病理诊断七剑下天山,医疗未来触手可及

一、病理诊断困局&#xff1a;刀尖上的医学艺术 1.1 金标准背后的隐痛 病理诊断被誉为"诊断的诊断"&#xff0c;医生需通过显微镜观察组织切片&#xff0c;在细胞迷宫中捕捉癌变信号。某省病理质控报告显示&#xff0c;基层医院误诊率达12%-15%&#xff0c;专家会诊…...

Python+ZeroMQ实战:智能车辆状态监控与模拟模式自动切换

目录 关键点 技术实现1 技术实现2 摘要&#xff1a; 本文将介绍如何利用Python和ZeroMQ消息队列构建一个智能车辆状态监控系统。系统能够根据时间策略自动切换驾驶模式&#xff08;自动驾驶、人工驾驶、远程驾驶、主动安全&#xff09;&#xff0c;并通过实时消息推送更新车…...

在 Spring Boot 项目里,MYSQL中json类型字段使用

前言&#xff1a; 因为程序特殊需求导致&#xff0c;需要mysql数据库存储json类型数据&#xff0c;因此记录一下使用流程 1.java实体中新增字段 private List<User> users 2.增加mybatis-plus注解 TableField(typeHandler FastjsonTypeHandler.class) private Lis…...