传送门

Description

有一副nm的地图,有nm块地,每块是下列四种中的一种:

墙:用#表示,墙有4个面,分别是前面,后面,左面,右面。

起点:用C表示,为主角的起点,是一片空地。

终点:用F表示,为主角的目的地,是一片空地。

空地:用 . 表示。

其中除了墙不能穿过,其他地方都能走。

主角有以下3种操作:

1.移动到相邻的前后左右的地方,花费一个单位时间。

2.向前后左右其中一个方向发射子弹,子弹沿直线穿过,打在最近的一堵墙的一面,然后墙的这面就会形成一个开口通往秘密通道。同一时间最多只能有两个开口,若出现有3个开口,出现时间最早的开口会立即消失。该操作不用时间。

3.可以从一个与开口相邻的空地跳进去,进入秘密通道,从另外一个开口正对的空地跳出来。这个过程花费一个单位时间。

地图四周都是墙,问主角最少用多少时间从C走到F。C和F

只会出现一次。

Input

第一行输入两个正整数n,m。

接下来n行,每行m个字符描述地图。

Output

输出1个整数,表示最短时间完成路途。如果无解输出nemoguce

Sample Input

Input 1
4 4
####
#.F#
#C.#
####
Input 2
6 8
########
#.##..F#
#C.##..#
#..#...#
#.....##
########
Input 3
4 5
#####
#C#.#
###F#
#####

Sample Output

Output 1

2

Output 2

4

Output 3

nemoguce

Data Constraint

对于50%的数据,4≤ n,m≤ 15。

对于100%的数据,4≤ n,m≤ 500。

Hint

总共用到8次操作,时间之和为4。如下图所示

1.向左射一枪,在(3,1)的右面出现开口。

2.向下射一枪,在(6,2)的上面出现开口。

3.向左从(3,1)进入秘密通道,从(6,2)中出来,到达(5,2)。用1单位时间。

4.向右射一枪,在(5,7)的左面出现开口,(3,1)右面的开口消失。

5.走进(6,2)的开口,出来到(5,6)。用1单位时间。

6.向上射一枪,在(1,6)的下面出现开口。

7.经过秘密通道,走到(2,6)。用1单位时间。

8.走到终点。用1单位时间。

Solution

每个不是墙的点向四周连边,同时向四周的墙前的点,连一条边权为 到最近墙距离+1的边

Code

//By Menteur_Hxy
#include<queue>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define M(a,b) memset(a,(b),sizeof(a))
#define F(i,a,b) for(register int i=(a);i<=(b);i++)
#define R(i,a,b) for(register int i=(b);i>=(a);i--)
#define E(i,u) for(register int i=head[u];i;i=nxt[i])
//#define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<26,stdin),p1==p2)?EOF:*p1++)
using namespace std;
typedef pair<int,int> PII; char buf[1<<26],*p1=buf,*p2=buf;
int read() {
int x=0,f=1; char c=getchar();
while(!isdigit(c)) {if(c=='-')f=-f; c=getchar();}
while(isdigit(c)) x=(x<<1)+(x<<3)+c-48,c=getchar();
return x*f;
} const int INF=0x3f3f3f3f;
const int N=510,N2=2510,N4=6250010;
int mv[5]={0,1,0,-1,0};
bool mp[N][N];
int n,m,tot,cnt,S,T;
int nxt[N4],to[N4],w[N4],head[N4],inq[N4],dis[N4];
PII wal[4]; int nd(int i,int j) {return i*m+j-m;}
void add(int a,int b,int c) {nxt[++cnt]=head[a],to[cnt]=b,w[cnt]=c,head[a]=cnt;} void search(int x,int y) {
int mi=INF;
// printf("%d--%d\n",x,y);
F(i,0,3) {
int xx=x+mv[i],yy=y+mv[i+1],len=1;
while(!mp[xx][yy]) xx+=mv[i],yy+=mv[i+1],len++;
wal[i]=PII(xx-mv[i],yy-mv[i+1]);
mi=min(mi,len);
// printf("%d %d\n",xx,yy);
}
// if(cnt>=37300) printf("%d %d\n",wal[0].first,wal[0].second);
F(i,0,3) add(nd(x,y),nd(wal[i].first,wal[i].second),mi);
} queue <int> Q;
void SPFA() {
M(dis,0x3f);
Q.push(S);dis[S]=0;
while(!Q.empty()) {
int u=Q.front(); Q.pop(); inq[u]=0;
E(i,u) {
int v=to[i];
// cout<<v<<endl;
if(dis[v]>dis[u]+w[i]) {
dis[v]=dis[u]+w[i];
if(!inq[v]) Q.push(v),inq[v]=1;
}
}
}
} int main() {
freopen("portal.in","r",stdin);
freopen("portal.out","w",stdout);
n=read(),m=read();
F(i,1,n) {
scanf("\n");
F(j,1,m) { char c=getchar();
if(c=='#') mp[i][j]=1;
else {
if(c=='C') S=nd(i,j);
if(c=='F') T=nd(i,j);
}
}
}
F(i,1,n) F(j,1,m) if(!mp[i][j]) search(i,j);
F(i,1,n) F(j,1,m) if(!mp[i][j])
F(k,0,3) if(!mp[i+mv[k]][j+mv[k+1]])
add(nd(i,j),nd(i+mv[k],j+mv[k+1]),1);
SPFA();
if(dis[T]==INF) return printf("nemoguce"),0;
else return printf("%d",dis[T]),0;
}

[jzoj 5781]【NOIP提高A组模拟2018.8.8】秘密通道 (最短路)的更多相关文章

  1. JZOJ 5818. 【NOIP提高A组模拟2018.8.15】 做运动

    5818. [NOIP提高A组模拟2018.8.15] 做运动 (File IO): input:running.in output:running.out Time Limits: 2000 ms  ...

  2. JZOJ 5812. 【NOIP提高A组模拟2018.8.14】 区间

    5812. [NOIP提高A组模拟2018.8.14] 区间 (File IO): input:range.in output:range.out Time Limits: 1000 ms  Memo ...

  3. 5820. 【NOIP提高A组模拟2018.8.16】 非法输入(模拟,字符串)

    5820. [NOIP提高A组模拟2018.8.16] 非法输入 (File IO): input:aplusb.in output:aplusb.out Time Limits: 1000 ms   ...

  4. [jzoj 5782]【NOIP提高A组模拟2018.8.8】 城市猎人 (并查集按秩合并+复杂度分析)

    传送门 Description 有n个城市,标号为1到n,修建道路花费m天,第i天时,若gcd(a,b)=m-i+1,则标号为a的城市和标号为b的城市会建好一条直接相连的道路,有多次询问,每次询问某两 ...

  5. [jzoj 5778]【NOIP提高A组模拟2018.8.8】没有硝烟的战争 (博弈论+dp)

    传送门 Description 被污染的灰灰草原上有羊和狼.有N只动物围成一圈,每只动物是羊或狼. 该游戏从其中的一只动物开始,报出[1,K]区间的整数,若上一只动物报出的数是x,下一只动物可以报[x ...

  6. [JZOJ5817] 【NOIP提高A组模拟2018.8.15】 抄代码

    Description J 君是机房的红太阳,每次模拟她总是 AK 虐场.然而在 NOIP2117 中,居然出现了另一位 AK 的选手 C 君! 这引起了组委会的怀疑,组委会认为 C 君有抄袭 J 君 ...

  7. [JZOJ5818] 【NOIP提高A组模拟2018.8.15】 做运动

    Description 一天,Y 君在测量体重的时候惊讶的发现,由于常年坐在电脑前认真学习,她的体重有了突 飞猛进的增长. 幸好 Y 君现在退役了,她有大量的时间来做运动,她决定每天从教学楼跑到食堂来 ...

  8. 【NOIP提高A组模拟2018.8.14】 区间

    区间加:差分数组修改 O(n)扫描,负数位置单调不减 #include<iostream> #include<cstring> #include<cstdio> # ...

  9. [JZOJ5781]【NOIP提高A组模拟2018.8.8】秘密通道

    Description 有一副n*m的地图,有n*m块地,每块是下列四种中的一种:墙:用#表示,墙有4个面,分别是前面,后面,左面,右面.起点:用C表示,为主角的起点,是一片空地.终点:用F表示,为主 ...

随机推荐

  1. Mybatis+0+null,小问题引发的血案

    Mybatis在进行<if test="status != null and status != ''">判空操作时,假设status为0的时候,该推断条件的值为fal ...

  2. Latex 排版技巧 1——数学公式对齐

    在我们排版数学推导式时,非常多时候我们希望可以让公式的等号对齐 这样更接近人的数学推导习惯 例如以下图效果图 使用 begin{aligned} end{aligned}将所需对齐的数学公式代码块包起 ...

  3. 软件project文档中的数据库模型设计

    背景:软件project文档之<数据库设计说明书>的结构设计部分要明白规划出数据库的概念结构设计.逻辑结构设计.物理结构设计,就是设计数据库的概念模型.逻辑模型.物理模型.那么.何为数据库 ...

  4. C# Socket 您的主机中的软件中止了一个已建立的连接 An established connection was aborted by the software in your host machine

    http://tieba.baidu.com/p/3223234493 问题: 服务端在接收客户端数据的时候,抛了个异常出来:System.Net.Sockets.SocketException: 您 ...

  5. React Native - 认识与环境搭建

    01 传统开发的痛点 1.人员稀缺 2.开发成本高 3.代码复用率低 4.无法动态更新 02 React Native的优点 1.跨平台 2.性能高 3.低投入 4.支持动态更新 03 开发环境搭建 ...

  6. WinForm c# 备份 还原 数据库(Yc那些事儿 转)

    Yc那些事儿 我愿意 为了我的幸福 奋斗终生     2008-11-17 18:04 WinForm c# 备份 还原 数据库 其实是个非常简单的问题,一个Form,一个Button,一个OpenF ...

  7. 1.ArcGis几何图形之几何计算

    /// <summary> /// 检测几何图形A是否包含几何图形B /// </summary> /// <param name="pGeometryA&qu ...

  8. cocos2d-x https

    cocos2d-x :2.1.3HttpClient.cpp文件中  bool configureCURL(CURL *handle)后边添加如下两句: curl_easy_setopt(handle ...

  9. SQLServer2008 关于数据转换

    全进位 select cast(ceiling(2.1111) as dec(18,0)) 结果:3

  10. javascript 核心概念(1)-数据类型

    语法 (1)到现在为止,大多数浏览器也还是支持到ECMAScript 第三版的标准. 核心概念就是一个语言的基本工作原理,涉及语法,操作符,数据类型. (2)javascript的一切--变量,函数名 ...