Tempter of the Bone

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)

Problem Description
The doggie found a bone in an ancient maze, which fascinated him a lot. However, when he picked it up, the maze began to shake, and the doggie could feel the ground sinking. He realized that the bone was a trap, and he tried desperately to get out of this maze.

The maze was a rectangle with sizes N by M. There was a door in the maze. At the beginning, the door was closed and it would open at the T-th second for a short period of time (less than 1 second). Therefore the doggie had to arrive at the door on exactly the T-th second. In every second, he could move one block to one of the upper, lower, left and right neighboring blocks. Once he entered a block, the ground of this block would start to sink and disappear in the next second. He could not stay at one block for more than one second, nor could he move into a visited block. Can the poor doggie survive? Please help him.

 
Input
The input consists of multiple test cases. The first line of each test case contains three integers N, M, and T (1 < N, M < 7; 0 < T < 50), which denote the sizes of the maze and the time at which the door will open, respectively. The next N lines give the maze layout, with each line containing M characters. A character is one of the following:

'X': a block of wall, which the doggie cannot enter; 
'S': the start point of the doggie; 
'D': the Door; or
'.': an empty block.

The input is terminated with three 0's. This test case is not to be processed.

 
Output
For each test case, print in one line "YES" if the doggie can survive, or "NO" otherwise.
 
Sample Input
4 4 5
S.X.
..X.
..XD
....
3 4 5
S.X.
..X.
...D
0 0 0
 
Sample Output
NO
YES
 
题目大意:
      输入一个n*m的迷宫,和一个时间t(代表可以在迷宫中生存的最长时间,可以理解为超过时间t迷宫将会坍塌,人也会死亡),
      迷宫有一个出口,这个出口只会在第t秒的时候打开,并且人每走一步,走过的道路就会坍塌,迷宫中还有一些墙,人是不能通过的。
      问:一个人从起始地点能否刚好在第t秒的时候到达出口。
 
解题思路:
      总体思路为 dfs+剪枝  因为最后问能否刚好在第t秒到达出口,所以需要搜索所有能够走的路程,寻找是否有满足条件的情况。
      但是深搜特别费时间,直接暴力搜索一定是会超时的,所以需要通过剪枝来缩短搜索的时间。
 
      首先想到的第一个剪枝是:当剩余的最小步数 > 剩余的时间时 最后是无法在第t秒到达出口的。
      接下来需要用到的一个剪枝为:奇偶剪枝
 
奇偶剪枝:
      我们可以定义一个map数组来表示每个点的奇偶性(每个点的奇偶性为该点横纵坐标的和)
      0 1 0 1 0 1 
      1 0 1 0 1 0

      0 1 0 1 0 1 
      1 0 1 0 1 0 
      0 1 0 1 0 1 
      有这个表可以发现 从为0的格子走一步,必然走向为1的格子;同样从为1的格子走一步,必然走向为0的格子.
      即 0-->1或1-->0 必然是奇数步       0-->0或1-->1 必然是偶数步 
      so~ 当遇到从0走到0但是要求奇数步、或者从1走到0但是要求偶数步,都可以之间判断不能到达。
      应用到这个题上就是:判断剩余时间和剩余的最小步数的奇偶性是否相同。相同则有可能到达、不同则一定不能到达。
      (奇数-偶数 = 奇数    奇数-奇数= 偶数    偶数-偶数=偶数)
 
AC代码:
 #include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <string.h>
#include <algorithm>
using namespace std; char p[][];
int a[][] = {-,,,,,,,-}; // 方向数组
int flag; // 标记是否能够在第t秒时到达出口
int n,m,t;
int sx,sy,dx,dy; // 起始地点 和 出口的地点 void dfs(int x,int y,int k)
{
int i;
if (x< || y< || x>n || y>m) // 判断是否超出边界
return ;
if (k == &&x == dx && y==dy) // 如果刚好在第t秒时到达出口 flag = 1
{
flag = ;
return ;
}
int tmp = k - abs(x-dx)-abs(y-dy);
if (tmp < || tmp&) // 判断 1、是否剩余的最小步数 > 剩余的时间 2、剩余时间和剩余的最小步数的奇偶性是否相同
return ; for (i = ; i < ; i ++)
{
if (p[x+a[i][]][y+a[i][]] != 'X') // 如果时墙则不能通过
{
p[x+a[i][]][y+a[i][]] = 'X'; // 走过之后路就坍塌了(之后不能在通过)
dfs(x+a[i][],y+a[i][],k-);
if (flag) return ; // 如果满足条件直接退出
p[x+a[i][]][y+a[i][]] = '.'; // 当前搜的这条路可能不满足条件 下次再搜别的路时要恢复原样
}
}
return ;
}
int main ()
{
int i,j;
while (scanf("%d%d%d",&n,&m,&t)!=EOF)
{
getchar();
int sum = ;
if (n==&&m==&&t==)
break;
for (i = ; i <= n; i ++)
{
for (j = ; j <= m; j ++)
{
scanf("%c",&p[i][j]);
if (p[i][j] == 'S'){sx = i; sy = j;}
else if (p[i][j] == 'D'){dx = i; dy = j;}
else if (p[i][j] == 'X') sum ++;
}
getchar();
} if (n*m-sum <= t) // 如果可以走的路的步数小于t 直接pass
{
printf("NO\n");
continue;
}
p[sx][sy] = 'X'; // 出发之后不能再回到起点
flag = ;
dfs(sx,sy,t);
if (flag)
printf("YES\n");
else
printf("NO\n");
}
return ;
}     
 

最新文章

  1. Centos 7 安装 和 卸载 Mysql5.7(压缩包)
  2. mvc与三层结构
  3. 【转】 StringUtils中 isNotEmpty 和isNotBlank的区别
  4. JAVA多线程与多进程
  5. 给eclipse装一些插件
  6. Json序列反序列类型处理帮助类
  7. 2.7. 属性的各种设置选项(Core Data 应用程序实践指南)
  8. 13 用Css做下拉菜单
  9. StringBulider简单用法
  10. Qt 5.9.4 如何静态编译和部署?
  11. js回溯法计算最佳旅行线路
  12. Python语言中的按位运算
  13. 机器学习之KNN算法
  14. MVC 打印解决方案--SNF快速开发平台3.1
  15. English trip V1 - 24. Accommodations Teacher:Maple Key: make suggestions 提出建议
  16. HDU 3746 - Cyclic Nacklace &amp; HDU 1358 - Period - [KMP求最小循环节]
  17. Matchvs 使用记录
  18. [HAOI2015]按位或
  19. 通过TodoList案例对比Vue.js的MVVM设计模式与JQuery的MVP设计模式
  20. md5.digest()与md5.hexdigest()之间的区别及转换

热门文章

  1. 04. prosition 的值都有哪些,其最本质的区别在哪里?
  2. centos的 / ~ - 的意思
  3. HDU5558(后缀自动机~在本串中找前一最长子相同串)
  4. springcloud(三)-Eureka
  5. 线性代数与simplex
  6. 对java的理解
  7. js中this那些事儿
  8. es put mapping
  9. cadence help文件库调出指令 :cdnshelp
  10. Docker搭建tomcat运行环境(Dockerfile方式)