在网上找到的解题算法:
这是一道经典搜索题,常规方法是深搜,当然要加上必要的剪枝条件
主要的剪枝条件有:
1、剩余可走区域小于时间
2、奇偶性剪枝
3、越界
4、超时等
下面主要说说奇偶性剪枝
若有一迷宫,将迷宫的每一个位置有0或1表示(x+y为偶数时 为0 否则为1):
0 1 0 1 0
1 0 1 0 1
0 1 0 1 0
1 0 1 0 1
从图中我们可以很清晰的看出:任意一个位置周围相邻的必然是与本身值相反的值,也就是说,要想走到与本身值相同的点必然要走偶数步;
同理,要想走到与本身相异的值的点必然要走奇数步;
所以,当两个位置的奇偶性相同时(同为0或同为1)若时间为奇数 则必然无法到达;当两个位置的奇偶性不同时(一个为1,另一个为0)若时间为偶数也必不能到达。
根据这个算法,可以实现成功的查找:
#if 1
#include <iostream>
#include <cmath>
using namespace std;
char ch[9][9];
int Find(int T, int Di, int Dj, int Si, int Sj)
{
/*如果想加为奇数,则不可能成功*/
if((Di + Dj + Si + Sj + T)%2 != 0)
return 0;
/*Dis是最短路径, 如果最短路径的时间都比所给时间长,那肯定超时*/
int Dis = abs(Si - Di) + abs(Sj - Dj);
if(Dis>T)
return 0;
/*查询到 'D' 所在的位置,即 Si == Di, Sj == Dj; 且时间正好是开门时间,则成功*/
if(Dis == 0 && T == 0)
return 1;
char temp;
temp = ch[Si][Sj];
ch[Si][Sj] = 'X';
if(ch[Si][Sj-1] != 'X' && Find(T-1, Di,Dj,Si,Sj-1))
return 1;
if(ch[Si][Sj+1] != 'X' && Find(T-1, Di,Dj,Si,Sj+1))
return 1;
if(ch[Si-1][Sj] != 'X' && Find(T-1, Di, Dj, Si-1, Sj))
return 1;
if(ch[Si+1][Sj] != 'X' && Find(T-1, Di, Dj, Si+1, Sj))
return 1;
ch[Si][Sj] = temp;
return 0;
}
int main()
{
int Si, Sj;
int Di, Dj;
int N,M,T;
int i,j,k;
int result;
while(cin>>N>>M>>T &&(N != 0 && M != 0 && T != 0))
{
/*保证所有的边界为 'X',这样数组就不会越界检查*/
for(i = 0; i <= N+1; i++)
for(j = 0; j <= M+1; j++)
ch[i][j] = 'X';
for(i = 1; i <= N; i++)
for(j = 1; j <= M; j++)
{
cin>>ch[i][j];
if(ch[i][j] == 'S')
{
Si = i;
Sj = j;
}
if(ch[i][j] == 'D')
{
Di = i;
Dj = j;
}
}
result = Find(T,Di, Dj, Si, Sj);
if(result == 1)
cout<<"YES"<<endl;
else
cout<<"NO"<<endl;
}
return 0;
}
#endif