马的遍历
题目描述
有一个 n × m n \times m n×m 的棋盘,在某个点 ( x , y ) (x, y) (x,y) 上有一个马,要求你计算出马到达棋盘上任意一个点最少要走几步。
输入格式
输入只有一行四个整数,分别为 n , m , x , y n, m, x, y n,m,x,y。
输出格式
一个 n × m n \times m n×m 的矩阵,代表马到达某个点最少要走几步(左对齐,宽 5 5 5 格,不能到达则输出 − 1 -1 −1)。
样例 #1
样例输入 #1
3 3 1 1
样例输出 #1
0 3 2
3 -1 1
2 1 4
提示
数据规模与约定
对于全部的测试点,保证 1 ≤ x ≤ n ≤ 400 1 \leq x \leq n \leq 400 1≤x≤n≤400, 1 ≤ y ≤ m ≤ 400 1 \leq y \leq m \leq 400 1≤y≤m≤400。
题目分析
这是一道经典的广度优先搜索(bfs)题,要是用深度优先(dfs)的话,会有很多点被重复走过,所以很多样例都超时。广度优先的话则不同,它把每个结点可能的下一步都搜索一遍,所以如果找到解时,那么这个解就是最优解,即花费了最少的搜索次数。这时我们可以使用队列(queue,一种先进先出的数据结构),将每个节点的下一步都插进(push)队列,然后将该节点弹出(pop),再规定一下插入的条件,就可以再队列为空之前找到所有的解。
代码实现
#include <bits/stdc++.h>
#include <queue> //包含头文件queue
using namespace std;
int cb[405][405];
bool vis[405][405]; //判断坐标是否已经来过,避免在某一个点死循环
int n,m,x,y;
//将马可能走的位置用数组存储起来,避免了麻烦地使用8个if
int dx[8] = {2,-2,2,-2,-1,1,-1,1};
int dy[8] = {1,1,-1,-1,2,2,-2,-2};
queue<pair<int,int>> q;//将坐标用pari存储,也可以自己定义一个坐标的结构体
int main(){
cin >> n >> m >> x >> y;
memset(cb,-1,sizeof(cb));//将所有坐标的最小步数初始化为-1
cb[x][y] = 0;
vis[x][y] = 1; //起始点已经来过
q.push(make_pair(x,y));
while(!q.empty()){
int x1 = q.front().first,y1 = q.front().second;
q.pop();
for (int i = 0; i < 8; ++i) {
int x2 = x1+dx[i];
int y2 = y1+dy[i];
//判断条件 下一个点不超过边界 且 没有遍历过
if (x2>=1 && x2<=n && y2>=1 && y2<=m && vis[x2][y2]==0){
q.push(make_pair(x2,y2));
vis[x2][y2] = 1;
cb[x2][y2] = cb[x1][y1] + 1;//步数加一
}
}
}
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
cout << left << setw(5) << cb[i][j]; //输出,注意格式
}
cout << endl;
}
}