nyoj999 师傅又被妖怪抓走了 (预处理+bfs+状态压缩)
题目999
题目信息
执行结果
本题排行
讨论区
师傅又被妖怪抓走了
时间限制:1000 ms | 内存限制:65535 KB
难度:3
描写叙述
话说唐僧复得了孙行者,师徒们一心同体,共诣西方。自宝象国救了公主,承君臣送出城西。沿路饥餐渴饮,悟空便为师傅去化斋。等悟空回来,悟净慌慌张张的对悟空说:“不好了,不好了”,还没等悟净说完,悟空说:“师傅又被妖怪抓走了”,悟净:“NO!” ,悟空一脸茫然。悟净:“师傅和二师兄都被妖怪抓走了”。悟空(晕!
)。
为了防止悟空救人,妖怪先把唐憎和八戒分别藏起来,如果悟空在T分钟之后还没找到人。那必然是被妖怪吃掉了。如果悟空在一个n行m列的矩阵内,悟空在每一分钟能够走到上,下,左,右的当中的一个能够走的位置,每次仅仅能走一步。
我们把发现定义为能够直接看到对方,也就是说两个人在同一行或者同一列,而且中间没有障碍物或者没有其它人就能够看到对方。
输入
有多组測试数据,每组首先是三个正整数n , m (3<=n,m<=100), T,(0<=T<=100) 分别代表行数,列数,规定的时间。接下来n 行,每行 m 个字符。
当中’ S ’ 代表悟空的位置,’ D ’代表师傅位置,’ E ’代表八戒的位置。而且保证都仅仅有一个. ’ X ’代表墙 ,’ . ’代表空地 .
输出
每组先输出一行Case c:(c表示当前的组数。从1開始计数);
接下来一行。假设悟空能够在规定时间内找到两人,则输出最少须要的时间,否则输出-1。
例子输入
5 6 3
XXD...
....E.
....X.
....S.
......
5 6 3
XDX...
....E.
......
....S.
......
5 6 8
XXDX..
.XEX..
......
....S.
......
例子输出
Case 1:
-1
Case 2:
3
Case 3:
-1
这道题让我非常焦灼啊。。。开个玩笑 哈哈
这道题的发现定义为能够直接看到对方。所以是不能走X D E 仅仅能是看到
所以在录入数据的时候 我们就提前做好预处理。
把D E的四个方向 能看到的地方都处理一下
然后就是bfs了 而在这里 我提交了几次 发现都不正确 后来无意中的一个数据启示了我
2 3 1
.S.
DXE
输出-1
2 3 3
.S.
DXE
输出3
聪明的你 看出来了吗?假设我们用通常的vis数组标记走过的路径
,对于第二条測试数据 仅仅会输出-1 由于它不会走回头路 。我又存在侥幸的心理 不用vis数组标记了 发现 果然TLE
然后就是不做了 走在路上还在想 到底在这里该怎么优化 。。。果然 睡觉的时候被我想到了 今天早上一大早 就起来做了。
详细方法是 假设又定义了两个数组visE【x】【y】 visD【x】【y】分别记录在x,y点 是否已经找到过E 或D 。
当某个点已经遍历过
我们再推断当前的visD 和visE和之前的是否一致 假设一致 我们就不用走 假设不一致 证明我们在其他点找到了D或E 如今要回来。
详细看代码把
#include <stdio.h>
#include <string.h>
#include <queue>
#include <algorithm>
using namespace std;
int n,m;
int t;
int s1,s2;
int d1,d2;
int e1,e2;
int dir[4][2]={1,0,0,1,-1,0,0,-1};
char map[105][105];
bool visD[105][105];
bool visE[105][105];
bool vis[105][105];
struct node
{
int x,y;
int step;
bool isfindD;
bool isfindE;
friend bool operator<(node a,node b)
{
return a.step>b.step;
}
};
void prepare(int x,int y,char ch)
{
char ch2;
int x2,y2;
if(ch=='D')
ch2='E',x2=e1,y2=e2;
else
ch2='D',x2=d1,y2=d2;
for(int i=x-1;i>=0;i--)
{
if(map[i][y]=='X'||i==x2&&y==y2)
break;
if(map[i][y]=='.')
map[i][y]=ch;
else if(map[i][y]==ch2)
map[i][y]='*';
else if(map[i][y]=='S')
{
map[i][y]=ch;
break;
}
}
for(int i=x+1;i<n;i++)
{
if(map[i][y]=='X'||i==x2&&y==y2)
break;
if(map[i][y]=='.')
map[i][y]=ch;
else if(map[i][y]==ch2)
map[i][y]='*';
else if(map[i][y]=='S')
{
map[i][y]=ch;
break;
}
}
for(int i=y+1;i<m;i++)
{
if(map[x][i]=='X'||x==x2&&i==y2)
break;
if(map[x][i]=='.')
map[x][i]=ch;
else if(map[x][i]==ch2)
map[x][i]='*';
else if(map[x][i]=='S')
{
map[x][i]=ch;
break;
}
}
for(int i=y-1;i>=0;i--)
{
if(map[x][i]=='X'||x==x2&&i==y2)
break;
if(map[x][i]=='.')
map[x][i]=ch;
else if(map[x][i]==ch2)
map[x][i]='*';
else if(map[x][i]=='S')
{
map[x][i]=ch;
break;
}
}
}
bool limit(int x,int y)
{
if(x==d1&&y==d2)
return false;
if(x==e1&&y==e2)
return false;
if(x<0||y<0||x>=n||y>=m||map[x][y]=='X')
return false;
return true;
}
void check_result(node &temp1)
{
int x=temp1.x;
int y=temp1.y;
if(map[x][y]=='D')
temp1.isfindD=true;
else if(map[x][y]=='E')
temp1.isfindE=true;
else if(map[x][y]=='*')
temp1.isfindD=temp1.isfindE=true;
}
int bfs(int x1,int y1,int time)
{
node temp1,temp2;
priority_queue<node>s;
while(!s.empty())
s.pop();
temp1.x=x1;temp1.y=y1;temp1.step=0;
temp1.isfindD=temp1.isfindE=false;
check_result(temp1);
visE[x1][y1]=temp1.isfindE;
visD[x1][y1]=temp1.isfindD;
vis[x1][y1]=true;
s.push(temp1);
while(!s.empty())
{
temp1=temp2=s.top();
s.pop();
if(temp1.step>time)
return -1;
if(temp1.isfindD&&temp1.isfindE)
return temp1.step;
for(int i=0;i<4;i++)
{
int x1=temp1.x+dir[i][0];
int y1=temp1.y+dir[i][1];
if(limit(x1,y1)&&(!vis[x1][y1]||!(visE[x1][y1]==temp1.isfindE&&visD[x1][y1]==temp1.isfindD)))
{
temp1.x=x1;
temp1.y=y1;
temp1.step++;
check_result(temp1);
vis[x1][y1]=true;
visE[x1][y1]=temp1.isfindE;
visD[x1][y1]=temp1.isfindD;
s.push(temp1);
temp1=temp2;
}
}
}
return -1;
}
int main()
{
int ncase=1;
while(~scanf("%d %d %d",&n,&m,&t))
{
memset(map,0,sizeof(map));
for(int i=0;i<n;i++)
{
getchar();
for(int j=0;j<m;j++)
{
scanf("%c",&map[i][j]);
if(map[i][j]=='S')
{
s1=i;
s2=j;
}
if(map[i][j]=='D')
{
d1=i;
d2=j;
}
if(map[i][j]=='E')
{
e1=i;
e2=j;
}
}
}
prepare(d1,d2,'D');
prepare(e1,e2,'E');
memset(visD,false,sizeof(visD));
memset(visE,false,sizeof(visE));
memset(vis,false,sizeof(vis));
printf("Case %d:\n%d\n",ncase++,bfs(s1,s2,t));
}
}
nyoj999 师傅又被妖怪抓走了 (预处理+bfs+状态压缩)的更多相关文章
- NYOJ999 师傅又被妖怪抓走了
只记得当下的眼疼 , ok 各种数据也试了 , 就是 他娘的不对 , 我也是醉了 . 也是日了最野的狗 附上日了哮天犬的代码 , 这个题 先放放, 一段时间后再试试 , 明天开始状态压缩吧 .为期两天 ...
- nyoj 999——师傅又被妖怪抓走了——————【双广搜】
师傅又被妖怪抓走了 时间限制:1000 ms | 内存限制:65535 KB 难度:3 描述 话说唐僧复得了孙行者,师徒们一心同体,共诣西方.自宝象国救了公主,承君臣送出城西,沿路饥餐渴饮,悟 ...
- uva 1601 poj 3523 Morning after holloween 万圣节后的早晨 (经典搜索,双向bfs+预处理优化+状态压缩位运算)
这题数据大容易TLE 优化:预处理, 可以先枚举出5^3的状态然后判断合不合法,但是由于题目说了有很多墙壁,实际上没有那么多要转移的状态那么可以把底图抽出来,然后3个ghost在上面跑到时候就不必判断 ...
- openjudge 大师兄,师傅被妖怪抓走啦
描述 孙悟空听到沙僧大喊一句:“大师兄,师傅被妖怪抓走啦!”于是孙悟空直追白骨精而去.孙悟空在一条长度为L的森林小路上飞奔,上面有L+1个整点,依次为0,1,2……L.白骨精会使用一种大范围的攻击法术 ...
- shu_1548 悟空问题(大哥,主妖怪抓走的朋友!)
http://202.121.199.212/JudgeOnline/problem.php?cid=1078&pid=17 分析: 直接暴力了.. . 代码: #include <s ...
- Light OJ 1037 - Agent 47(预处理状态压缩DP)
题目大意: 有个特工要执行任务,他会遭遇到最多15个目标,特工必须把他们全部杀死.当他杀死一个目标后他可以使用目标的武器来杀死其他人.因此他必须有一个杀人的顺序,使得他开枪的次数最小. 现在给你一个表 ...
- poj3669 Meteor Shower(预处理+bfs)
https://vjudge.net/problem/POJ-3669 先给地图a[][]预处理每个位置被砸的最小时间.然后再bfs. 纯bfs,还被cin卡了下时间.. #include<io ...
- 【HDOJ3567】【预处理bfs+映射+康拓展开hash】
http://acm.hdu.edu.cn/showproblem.php?pid=3567 Eight II Time Limit: 4000/2000 MS (Java/Others) Me ...
- 【NOIP2013】 华容道 bfs预处理+bfs
这一题我们考虑一个最裸的算法: 我们设$dp[i][j][k][l]$表示当前棋子在$(i,j)$且空格在$(k,l)$时的最小步数 然后显然随便转移一下就好了,时间复杂度为$O(q(nm)^2)$. ...
随机推荐
- SWFupload多图片上传入门教程
本文为转载内容,但所讲内容亲身试验证明可用,转载过来希望能帮助到有需要的人. 转载地址:http://blog.csdn.net/kongjiea/article/details/24290373#c ...
- MySQL Field排序法
检索 id = 2 or id = 5 or id = 9 or id = 56 or id = 38.然后按照 2 , 5, 9, 56, 38 这个顺序排列,这是题目要求 以下为解决方案: 1 ...
- Python学习第一篇
好久没有来博客园了,今天开始写自己学习Python和Hadoop的学习笔记吧.今天写第一篇,Python学习,其他的环境部署都不说了,可以参考其他的博客. 今天根据MachineLearning里面的 ...
- linux 查看tomcat 实时日志
进入tomcat下logs文件夹下,若没有Catalina.out文件先去启动服务在回到logs文件夹输入 tail -f catalina.out ,可看到实时日志
- Java 学习(14):接口 & 包(设置 CLASSPATH 系统变量)
Java 接口(英文:Interface) 定义:在JAVA编程语言中,接口是一个抽象类型,是抽象方法的集合,接口通常以 interface 来声明. 一个类通过继承接口的方式,从而来继承接口的抽象方 ...
- 从头认识Spring-2.4 基于java的标准注解装配-@Inject(2)-通过set方法或者其它方法注入
这一章节我们来讨论一下基于java的标准注解装配标签@Inject是如何通过通过set方法或者其它方法注入? 在使用@Inject标签之前.我们须要在pom文件中面增加以下的代码: <depen ...
- 我的vim配置记录
一 总体介绍 配置路径,/etc/vim/vimrc,这个是系统的vim配置,假设一台PC多个用户使用,每一个用户的习惯不同的话,能够使用不同的配置.在用户文件夹下新建一个.vimrc的文件就能够了. ...
- python-打开网页
最近一直想通过python来实现网页的操作.因为想把自己vimrc的配置直接通过脚本来实现.比较理想的情况下,就是通过一个脚本,把自己需要的一些资源直接从网上下载下来,然后再对下载的资源进行安装等操作 ...
- jquery插件库http://www.jq22.com/
http://www.jq22.com/ http://www.jq22.com/jquery/%E5%8A%A0%E8%BD%BD http://www.jq22.com/jquery/%E5%BC ...
- HTTP -- 请求/响应 结构
一:一个HTTP请求报文由四个部分组成:请求行.请求头部.空行.请求数据. 1.请求行 1.请求方法:GET POST 2.URL字段 3.HTTP版本字段 2.请求头 1.Accept:浏览器可接受 ...