前言
🌟🌟本期讲解关于力扣的几篇题解的详细介绍~~~
🌈感兴趣的小伙伴看一看小编主页:GGBondlctrl-CSDN博客
🔥 你的点赞就是小编不断更新的最大动力
🎆那么废话不多说直接开整吧~~
目录
📚️1.BFS如何解决最短路问题
在解决下面几道题之前,小编要展开讲讲为啥我们的BFS宽度优先遍历可以解决这里的最短路问题,我们在之前已经了解到,这个的宽度优先遍历BFS的遍历方法如同病毒扩散的方式进行扩散遍历,那么好就是一个关键
且看这张图:
假如说,我们从A到我们的I,那么最短路径就是:
A C E F I
注意:BFS只适用于边权为一的情况;
那么BFS如何进行操作?
首先我们了解到BFS的扩散是一层一层的,那么我们就可以利用这个特性;
假如我们要找A到E的最短路径,那么如下图:
可以看到,我们与A连接的值就是BC两个节点,那么我们可以将BC看做一层,那么我们在以BC为一层,进行扩散,那么就会得到下一层就是DE,那么此时这一层就包含我们的目标节点,那么就可以知道我们到达E节点的步数了~~~
那么基于上述,我们重A到目标点I,的实例如下:
注意我们在遍历的过程中,已经被遍历的节点,那么对应就不能够进行遍历的操作了,那么在一层一层拨开后:
A层 (BC层) (DE层) (FG层)
那么这就是我们的BFS解决最短路问题;
📚️2.迷宫中离入口最近的出口
🚀2.1题目描述
给你一个 m x n
的迷宫矩阵 maze
(下标从 0 开始),矩阵中有空格子(用 '.'
表示)和墙(用 '+'
表示)。同时给你迷宫的入口 entrance
,用 entrance = [entrancerow, entrancecol]
表示你一开始所在格子的行和列。
每一步操作,你可以往 上,下,左 或者 右 移动一个格子。你不能进入墙所在的格子,你也不能离开迷宫。你的目标是找到离 entrance
最近 的出口。出口 的含义是 maze
边界 上的 空格子。entrance
格子 不算 出口。
请你返回从 entrance
到最近出口的最短路径的 步数 ,如果不存在这样的路径,请你返回 -1
。
如下:
就是我们会出生在entrance的地方,然后找到最短的出口,返回我们离这个出口的距离
其中 “+”为墙,不可以触碰,不可以穿过;
🚀2.2题目解析
到这里就很明显了,我们使用BFS的最短路径解决办法,以入口为起点,开始扩散,如果扩散的一层到了边缘,那么就可以进行计算返回最短路径了;
解题思路:
1.创建队列,创建一个参照数组来规定哪些地址我们已经遍历了
2.将我们此时的位置放入队列中,然后此位置标记
3.在队列不可为空的时候,我们要拿出一层的节点来进行扩散操作,所以需要拿到此时队列的长度;
4.条件判断,入队列即可
🚀2.3题目代码
代码如下所示:
class Solution {
int[] dx = { 0, 0, 1, -1 };
int[] dy = { 1, -1, 0, 0 };
public int nearestExit(char[][] maze, int[] entrance) {
int m = maze.length;
int n = maze[0].length;
boolean[][] vis = new boolean[m][n];
Queue<int[]> q = new LinkedList<>();
q.add(new int[] { entrance[0], entrance[1] });
vis[entrance[0]][entrance[1]] = true;
int step = 0;
while (!q.isEmpty()) {
step++;
int sz = q.size();
for (int i = 0; i < sz; i++) {
int[] index = q.poll();
int a = index[0];
int b = index[1];
for (int k = 0; k < 4; k++) {
int x = a + dx[k];
int y = b + dy[k];
if (x >= 0 && x < m && y >= 0 && y < n && vis[x][y] == false && maze[x][y] == '.') {
if(x == 0 || x == m - 1 || y == 0 || y == n - 1){
return step;
}
q.add(new int[] { x, y });
vis[x][y] = true;
}
}
}
}
return -1;
}
}
注意:我们在扩散的时候是上下左右四个方向的扩散 ,并且在BFS的时候,我们要将一层的节点拿出来进行扩散,再将这一层扩散的节点作为一下层来进行扩散,这就是为什么我们要计算队列的长度;
📚️3.最小基因变化
🚀3.1题目描述
基因序列可以表示为一条由 8 个字符组成的字符串,其中每个字符都是 'A'
、'C'
、'G'
和 'T'
之一。
假设我们需要调查从基因序列 start
变为 end
所发生的基因变化。一次基因变化就意味着这个基因序列中的一个字符发生了变化。
- 例如,
"AACCGGTT" --> "AACCGGTA"
就是一次基因变化。
另有一个基因库 bank
记录了所有有效的基因变化,只有基因库中的基因才是有效的基因序列。(变化后的基因必须位于基因库 bank
中)
给你两个基因序列 start
和 end
,以及一个基因库 bank
,请你找出并返回能够使 start
变化为 end
所需的最少变化次数。如果无法完成此基因变化,返回 -1
。
注意:起始基因序列 start
默认是有效的,但是它并不一定会出现在基因库中。
总结:
就是将每个单词进行变化(可以变成A C G T)其中的一个,并且每次变化只要在基因库中存在,那么就是有效的变化,所以要求的就是从一个序列变成另一个序列的最短路径
🚀3.2题目分析
我就以题目的列子来进行举例吧:
如下图所示:
上述就是整个分析过程:
1.我们要对于一个序列上的每一个字符进行替换,在满足存在bank的情况下,将对应的序列进行入队列(作为一层),若不满足那么就不管;
2. 每一次进行替换的时候,都要将同一层的序列进行替换的操作,就是一步;所以这里要利用队列的长度
3.在添加序列进入队列中的时候,也要将对应序列进行记录表示已经出现过了,不能变化回去
4.在满足没有遍历过,并且存在bank中时,添加进入队列后,要判断是否是目标序列,返回我们的步数(一个变量,每次一层搞完后,就加一即可)
🚀3.3题目代码
代码如下所示:
class Solution {
public int minMutation(String startGene, String endGene, String[] bank) {
//首先将我们的基因库存入哈希表中
Set<String> isBank = new HashSet<>();
for (String s : bank) {
//判断存在即可
isBank.add(s);
}
if(startGene.equals(endGene)){
return 0;
}
if(!isBank.contains(endGene)){
return -1;
}
Queue<String> q = new LinkedList<>();
int step = 0;
//定义一个哈希表来判断是否已经遍历过了
Map<String, Boolean> vis = new HashMap<>();
//两层循环
q.add(startGene);
vis.put(startGene, true);
while (!q.isEmpty()) {
step++;
int count = q.size();
for (int k = 0; k < count; k++) {
String str = q.poll();
for (int i = 0; i < str.length(); i++) {
//进行四次变化
for (int j = 0; j < 4; j++) {
char ch = 'R';
if(j == 0){
ch = 'A';
}else if(j == 1){
ch = 'C';
}else if(j == 2){
ch = 'G';
}else if(j == 3){
ch = 'T';
}
StringBuilder sb = new StringBuilder(str);
sb.setCharAt(i, ch);
String sb_str = sb.toString();
if(isBank.contains(sb_str) && !vis.containsKey(sb_str)){
if(sb_str.equals(endGene)){
return step;
}
q.add(sb_str);
vis.put(sb_str,true);
}
}
}
}
}
return -1;
}
}
其实大体的思路和上面的迷宫题目差不多,就是一层一层进行剥去,这里的核心剥去就是一队列长度为循环条件,直到一层出完,并扩散至另一层后才会开启下一层的扩散~~~
📚️4.总结
本期小编主要讲解了关于BFS如何解决最短路径的问题,其主要思想就是利用BFS进行层层扩散,并一层一层剥离的思想,来决定最短路径
1926. 迷宫中离入口最近的出口 - 力扣(LeetCode)
🌅🌅🌅~~~~最后希望与诸君共勉,共同进步!!!
💪💪💪以上就是本期内容了, 感兴趣的话,就关注小编吧。
😊😊 期待你的关注~~~