洛谷 P1443 马的遍历---详细题解

目录

  • 马的遍历
    • 题目描述
    • 输入格式
    • 输出格式
    • 样例 #1
      • 样例输入 #1
      • 样例输出 #1
    • 提示
      • 数据规模与约定
  • 题目分析
  • 代码实现

马的遍历

题目描述

有一个 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 1xn400 1 ≤ y ≤ m ≤ 400 1 \leq y \leq m \leq 400 1ym400

题目分析

这是一道经典的广度优先搜索(bfs)题,要是用深度优先(dfs)的话,会有很多点被重复走过,所以很多样例都超时。广度优先的话则不同,它把每个结点可能的下一步都搜索一遍,所以如果找到解时,那么这个解就是最优解,即花费了最少的搜索次数。这时我们可以使用队列(queue,一种先进先出的数据结构),将每个节点的下一步都插进(push)队列,然后将该节点弹出(pop),再规定一下插入的条件,就可以再队列为空之前找到所有的解。

代码实现

#include 
#include     //包含头文件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;
    }
}

你可能感兴趣的:(深度优先,图论,算法,蓝桥杯)