题目描述
给定一个n*m的二维整数数组,用来表示一个迷宫,数组中只包含0或1,其中0表示可以走的路,1表示不可通过的墙壁。
最初,有一个人位于左上角(1, 1)处,已知该人每次可以向上、下、左、右任意一个方向移动一个位置。
请问,该人从左上角移动至右下角(n, m)处,至少需要移动多少次。
数据保证(1, 1)处和(n, m)处的数字为0,且一定至少存在一条通路。
输入格式
第一行包含两个整数n和m。
接下来n行,每行包含m个整数(0或1),表示完整的二维数组迷宫。
输出格式
输出一个整数,表示从左上角移动至右下角的最少移动次数。
数据范围
1≤n,m≤100
样例
输入样例:
5 5
0 1 0 0 0
0 1 0 1 0
0 0 0 0 0
0 1 1 1 0
0 0 0 1 0
输出样例:
8
数组模拟队列
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
const int N = 110;
typedef pair<int, int> PII;
int n, m;
int g[N][N];//存放地图
int d[N][N];//存 每一个点到起点的距离
PII q[N * N];//手写队列
int bfs()
{
int hh = 0, tt = 0;
q[0] = {0, 0};
memset(d, - 1, sizeof d);//距离初始化为- 1表示没有走过
d[0][0] = 0;//表示起点走过了
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};//x 方向的向量和 y 方向的向量组成的上、右、下、左
while(hh <= tt)//队列不空
{
PII t = q[hh ++ ];//取队头元素
for(int i = 0; i < 4; i ++ )//枚举4个方向
{
int x = t.first + dx[i], y = t.second + dy[i];//x表示沿着此方向走会走到哪个点
if(x >= 0 && x < n && y >= 0 && y < m && g[x][y] == 0 && d[x][y] == -1)//在边界内 并且是空地可以走 且之前没有走过
{
d[x][y] = d[t.first][t.second] + 1;//到起点的距离
q[ ++ tt ] = {x, y};//新坐标入队
}
}
}
return d[n - 1][m - 1]; //输出右下角点距起点的距离即可
}
int main()
{
cin >> n >> m;
for(int i = 0; i < n; i ++ )
for(int j = 0; j < m; j ++ )
cin >> g[i][j];
cout << bfs() << endl;
return 0;
}
C++ queue
#include <iostream>
#include <cstring>
#include <queue>
using namespace std;
const int N = 110;
typedef pair<int, int> PII;
int n, m;
int g[N][N], d[N][N];
int bfs()
{
queue< pair<int, int> > q;
q.push({0, 0});
memset(d, -1, sizeof(d));
d[0][0] = 0;
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
while (q.size())//队列不为空
{
PII t = q.front();//取队头元素
q.pop();//出队
for (int i = 0; i < 4; i++)
{
int x = t.first + dx[i], y = t.second + dy[i];
if (x >= 0 && x < n && y >= 0 && y < m && g[x][y] == 0 && d[x][y] == -1)
{
d[x][y] = d[t.first][t.second] + 1;//当前点到起点的距离
q.push({x, y});//将新坐标入队
}
}
}
return d[n - 1][m -1];
}
int main()
{
cin >> n >> m;
for (int i = 0; i < n; i++)
for (int j = 0; j < m; j++)
cin >> g[i][j];
cout << bfs() << endl;
return 0;
}
打印路径
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;
const int N = 110;
typedef pair<int, int> PII;
PII q[N*N],Prev[N][N];
int g[N][N], d[N][N];
int n, m;
int bfs()
{
int hh = 0, tt = 0;
q[0] = {0, 0};
memset(d, -1, sizeof d);
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
d[0][0] = 0;
while(hh <= tt)
{
PII t = q[hh ++ ];
for(int i = 0; i < 4; i ++ )
{
int x = dx[i] + t.first, y = t.second + dy[i];
if(x >= 0 && x < n && y >= 0 && y < m && g[x][y] == 0 && d[x][y] == -1)
{
d[x][y] = d[t.first][t.second] + 1;
Prev[x][y] = t;
q[++ tt] = {x, y};
}
}
}
int x = n - 1, y = m - 1;
while(x || y)//有一个不d等于0
{
cout << x << ' ' << y << endl;
PII t = Prev[x][y];
x = t.first, y = t.second;
}
return d[n - 1][m - 1];
}
int main()
{
cin >> n >> m;
for(int i = 0; i < n; i ++ )
for(int j = 0; j < m;j ++)
cin >> g[i][j];
cout << bfs() << endl;
return 0;
}
python
def bfs():
d[0][0] = 0
queue = [(0, 0)]
dx = [-1, 0, 1, 0]
dy = [0, 1, 0, -1]
while queue : #队列不为空
x, y = queue.pop(0)
for i in range(4):
a = x + dx[i];
b = y + dy[i];
if a >= 0 and a < n and b >= 0 and b < m and g[a][b] == 0 and d[a][b] == -1:
queue.append((a,b))#入队
d[a][b] = d[x][y] + 1
print(d[n - 1][m - 1])
#main
n, m = map(int, input().split())#map函数对分割输入后的字符列表转换成整型
g = [[-1 for j in range(m)] for i in range(n)] # 存储地图
for i in range(n):
in_li = list(map(int, input().split()))
for j in range(m):
g[i][j] = in_li[j];
d = [[-1 for i in range(m)] for j in range(n)]#初始化为 - 1
bfs()
附上打印路径的代码,如果对你有帮助,请留个评论或者点个赞,谢谢。
打印路径写的很清楚
就是因为评论区翻了翻没找到详细解释打印路径的,才发了这个评论哈哈哈
是不是GPT写的哈哈哈我今天也生成了差不多的代码
用GPT4写的,解释的很清楚,这就是科技的力量哈哈哈
为什么vector[HTML_REMOVED] q;就会报错
PII q[N * N];这样才可以啊,我不理解这个是为什么,原则上不都是可以的吗?
兄弟们 我想用res 代表走过的长度 没走一次不是都会离终点只进一格吗? 那么我在每个while循环当中 res++ 为什么答案不对???
样例能过
为什么我这个答案一直不对
你不会是
dx[4]={1,0-1,0}
这句里漏了个逗号吧…确实
为什么初始化队列hh=0,tt=0?
我记得以前y总代码tt是等于-1的,hh和tt初始化有什么原理吗
tt是初始-1,y总也习惯-1,但是这里是从起点开始q[0] = {0,0}; //此时默认将起点加入队列了
由于q[++ tt] = x;
所以tt变成了 -1 + 1 = 0 ,所以初始化的时候hh = 0, tt = 0; //队列中已经有一个元素了hh~
原来如此,要根据题目来初始化是吧?hhh感谢大佬答惑
因为队伍里有一个点了,根据需求来吧…我也刚开始学习加油hh~
一个很奇怪的问题。Y总讲输出路径的时候用的代码如下
但如果不定义临时变量t,变成
就会少输出(2,4)这个点。实在想不明白,求解
y = pre[x][y].second;里的x 会根据x = pre[x][y].first,改变所肯定会少几个点
确实如此,感谢
不是都一样吗,我等下回去试一试
啊啊啊啊啊你的问题解决了我找了好久好久的bug!!!Orz
不一样,下面那个,先改变了x,相当于后面的第一个下标变了
原来如此啊,出现了覆盖的问题😁
我爱你!!! ┭┮﹏┭┮,这注释救大命
写得太好了
ORZ
求大佬援助,这个代码为何不对?
#include [HTML_REMOVED]
using namespace std;
long long int n,sum=100000000,m,cup,equ[110][110],ans,ok[110][110],px[4]={0,-1,0,1},py[4]={1,0,-1,0};
bool edge(int x,int y){
return x>=1&&x<=n&&y>=1&&y<=m;
}
void bfs(int x,int y){
if(x==n&&y==m) sum=min(ans,sum);
ok[1][1]=-1;
for(int i=0;i<4;i){
if(edge(x+px[i],y+py[i])&&ok[x+px[i]][y+py[i]]==0){
cup=ans;
// cout<[HTML_REMOVED]>equ[i][j];
if(equ[i][j]==1) ok[i][j]=-5;
}
}
bfs(1,1);
cout<<sum;
return 0;
}
%%%%
请问为什么我这个代码不行呢?
当输入是:
10 10
0 0 0 0 0 0 1 0 0 1
0 0 1 0 0 1 0 0 0 1
0 0 1 0 0 1 0 0 0 0
0 0 0 0 0 0 0 0 0 0
1 1 1 0 0 0 0 1 0 0
0 0 0 0 0 0 1 0 0 1
0 0 0 1 1 1 0 1 1 1
0 0 0 0 0 0 0 0 0 1
0 0 0 0 1 0 1 0 0 0
0 0 0 0 1 0 1 1 1 0
我输入18,正确答案20
我只能说你小子很强, i也不 h也不 tt也不++
写的很好但是但凡你设置一下断点,就能看出大错误哥们
为什么pair数组(模拟队列的那个数组)要是N*N的长度呢
因为有数组维度NN,一共有NN 个元素啊
啊哦哦哦懂了谢谢
为什么每一次都要取走队头元素???是因为这个点已经走过了吗?
因为要一层层遍历啊,取队头元素能够保证在遍历下一层的时候把这一层已经走完了,这一层能够走到的点都入队了
为什么d要初始成-1,既然是bfs距离肯定是上一个+1的距离,所以只要初始化第一个点的距离为0就行了吧,剩下的点都是从0点开始+
中的
可以用来判断那个位置是不是第一次走
为什么用BFS找出的路径一定是最短的?
权值都一样,一层一层找肯定BFS最短
谢谢~
其实可以这样理解,如果有几条线路都可以到终点得话,这几条线路都是同样的速度,如果某条路线最先到达了,其他线路就不让到达了,相同的速度,先到的是不是距离最短。
因为每个位置在d数组也就是距离数组中能且只能被赋一次值,d数组中对应位置的值就是第一次找到这个元素所走的步数
既然是第一次找到这个点,肯定就是最短路径
现在输出路径是倒着输出的,那路径如果想正着输出该怎么办呢?
加个栈,先进后出,重新输出保存路径的栈就行。
用栈也可以,reverse 反转一下数组也可以
大哥写的最好
怎么好多人都是你这个头像呢
这是acwing网站初始的头像
soga
赞,您写得最清晰。