杭电ACM_1010_搜索题

在网上找到的解题算法:

这是一道经典搜索题,常规方法是深搜,当然要加上必要的剪枝条件

主要的剪枝条件有:

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


 

你可能感兴趣的:(杭电ACM_1010_搜索题)