传送门

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. C++ Primer 学习笔记_5_变量和基本类型(续2)

     变量和基本类型 七.枚举 枚举不但定义了整数常量集,并且还把它们聚集成组. 枚举与简单的const常量相比孰优孰劣, 通过以下一段代码. 一看便知: enum {input, output, a ...

  2. Swift基本常识点

    import Foundation // 单行注释 // 多行注释(支持嵌套,OC是不支持的) // 常量let,初始化之后就不可改变. // 常量的具体类型可以自动识别,等号后面是什么类型,它就是什 ...

  3. monitor weblogic server ,Very simple to use, weblogic监控、巡检、故障简单小工具

        1. 开发了一个简单的监视weblogic执行情况的小程序.各位朋友下载下来试试,不用登陆console就能够知道server的执行状况,包含了jvm.线程.jdbc.状态jms等:另一个更简 ...

  4. WPF学习笔记——在“System.Windows.StaticResourceExtension”上提供值时引发了异常

    在"System.Windows.StaticResourceExtension"上提供值时引发了异常 因应需要,写了一个转换器,然后窗体上引用,结果就出来这个错.编译的时候没事, ...

  5. yum install tomcat

    安装tomcat6 yum install tomcat6 tomcat6-webapps tomcat6-admin-webapps 启动tomcat6 service tomcat6 start ...

  6. Windows热键注册的底层原理

    要像系统注册一个全局热键,需要用到RegisterHotKey,函数用法如下(MSDN):BOOL RegisterHotKey(                                    ...

  7. Android中Calendar类的用法总结

    Calendar是Android开发中需要获取时间时必不可少的一个工具类,通过这个类可以获得的时间信息还是很丰富的,下面做一个总结,以后使用的时候就不用总是去翻书或者查资料了. 在获取时间之前要先获得 ...

  8. spring:利用Spring AOP 使日志输入与方法分离

    对方法进行日志输出是一种很常见的功能.传统的做法是把输出语句写在方法体的内部,在调用该方法时,用输入语句输出信息来记录方法的执行! 1.先写一个普通类: package com.importnew; ...

  9. [BZOJ 1735] Muddy Fields

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=1735 [算法] 二分图最小覆盖 [代码] #include<bits/stdc ...

  10. 【BZOJ 2252】 矩阵距离

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=2252 [算法] 将所有是”1“的点入队,然后广度优先搜索,即可 [代码] #incl ...