poj 2488 A Knight's Journey(DFS)

题目大意:给出一个国际棋盘的大小,判断马能否不重复的走过所有格,并记录下其中按字典序排列的第一种路径。

 

马的遍历是一道经典回溯题,当然还是DFS...这题有2个要密切注意的地方:

 

1、题目要求以字典序方式输出,要以字典序输出路径,那么方向数组就要以特殊的顺序排列了,这样只要每次从dfs(0,0)开始搜索,第一个成功遍历的路径一定是以字典序排列.

 

2、国际象棋的棋盘,横行为字母,表示横行坐标的是y;纵行为数字,表示纵行的坐标是x...一开始又搞反了...


#include <iostream>
#include <cmath>
#include <cstring>
#include <cstdio>
#include <queue>

using namespace std;

int dir[8][2]={-2,-1,-2,1,-1,-2,-1,2,1,-2,1,2,2,-1,2,1};   //记录方向
int g,a,b;//g用来记录是否找到解,找到后不再搜索
int vist[26][26],path[26][2];

void find(int i,int j,int k)//i,j是要走的格子,k记录已经走过的步数
{
    if(k==a*b)//走完了
    {
        for(int i=0;i<k;i++)
            printf("%c%d",path[i][0]+'A',path[i][1]+1);
        printf("\n");
        g=1;
    }
    else
    {
        for(int x=0;x<8;x++)//8个方向依次搜索
        {
            int n=i+dir[x][0];
            int m=j+dir[x][1];
            if(n>=0&&n<b&&m>=0&&m<a&&!vist[n][m]&&!g)
            {
                vist[n][m]=1;//标记已走
                path[k][0]=n,path[k][1]=m;
                find(n,m,k+1);
                vist[n][m]=0;//清除标记
            }
        }
    }
}

int main()
{
    int n;
    scanf("%d",&n);
    for(int m=0;m<n;m++)
    {
        g=0;
        scanf("%d %d",&a,&b);
        for(int i=0;i<a;i++) //初始化
            for(int j=0;j<b;j++)
                vist[i][j]=0;
        vist[0][0]=1;
        path[0][0]=0,path[0][1]=0;
        printf("Scenario #%d:\n",m+1);
        find(0,0,1);
        if(!g) printf("impossible\n");
        printf("\n");
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值