·为了分析方便,可以先做一个题目简化。去掉“钥匙”这个条件,那么就是一个BFS或者SPFA……现在加上该条件。如本题只给出最多两种钥匙,当然你可以继续坚持BFS等方式,时间不会太差。但是一旦钥匙种类上升至15的时候,就有太多情况需要处理(光说你写BFS的if就是很长的过程,但个人认为时间复杂度依旧能过这道题)。

·如图是简化版本的决策方式(为与后文呼应,用SPFA):

大方块是整个地图。小方块是一个房间。那么你可以在向四个方向走,前提是有路可走(没有墙)。你本可以轻松拯救大兵瑞恩,然后过上幸福快乐的生活,但是你可能在实际问题中遭遇这样的绝望情况:

       明明有路可走,但是那扇门你没有钥匙。现在你开始幻想,怎样才能走过去呢……你妄想:要是在这之前有开这扇门的钥匙就好了,那我同样是走相同的路来到这扇门前,但是在那个时空的我就可以轻松开门,这个时空的我就只能绝望地去他地寻找钥匙。

·我们谈到了时空断裂。那么就干脆把每个时空也作为状态的一部分吧!(注意,在这之前你的状态只有一个元素,就是当前点坐标)。

·由图可见,你要时空穿梭的条件是你必须具备相应的钥匙。图中:如果你在这个房间里捡到了B,C钥匙,那么就有这种穿越方式(蓝色有向边)。由于在这里“时空”与“坐标位置”的地位是相同的【它们无制约关系】。也就是说你穿越两次时空在向左走一步,和将这些动作反过来,在可行的情况下是等价的。说明这一点的目的在于:这启示我们,其实这图中6条边的意义是相同的。

         这样一来,这道题就变成了普通的最短路问题,只是路多了些,判断条件多了些。所以可以用二维状态来表示当前情况,最终答案在每个时空里找一个最优的即可(没喊你把所有钥匙拿完啊!)。

 1 #include<bits/stdc++.h>
2 #define go(i,a,b) for(int i=a;i<=b;i++)
3 #define fo(i,a,x) for(int i=a[x];i>-1;i=e[i].next)
4 #define mem(a,b) memset(a,b,sizeof(a))
5 #define P 3000
6 #define inf 0x3f3f3f3f
7 using namespace std;const int N=203
;
8 int dx[4]={0,0,1,-1},dy[4]={1,-1,0,0
};
9 int n,m,p,f,S[N][N],id[N][N],key[N],head[N],k=0
,s,t,dis[P][N];
10 struct E{int v,next,rar;}e[N*N*4];bool
inq[P][N];
11 void ADD(int u,int v,int rar){e[k]=(E){v,head[u],rar};head[u]=k++
;}
12 struct G{int
x,y;};
13 int
spfa()
14
{
15 mem(dis,0x3f);dis[1][1]=0;inq[1][1]=1
;
16 queue<G>q;q.push((G){1,1
});
17 while(!
q.empty())
18
{
19 G U=q.front();q.pop();inq[U.x][U.y]=0
;
20 dis[U.x|key[U.y]][U.y]=min(dis[U.x|
key[U.y]][U.y],dis[U.x][U.y]);
21 U.x|=
key[U.y];
22
23 fo(i,head,U.y)if(U.x%(e[i].rar<<1)>=
e[i].rar)
24
{
25 int v=
e[i].v;
26 if(dis[U.x][v]>dis[U.x][U.y]+1
)
27
{
28 dis[U.x][v]=dis[U.x][U.y]+1
;
29 if(!
inq[U.x][v])
30
{
31 inq[U.x][v]=1
;
32
q.push((G){U.x,v});
33
}
34
}
35
}
36
}
37 int ret=inf;go(i,1,P-1)ret=
min(ret,dis[i][t]);
38 if(ret==inf)ret=-1;return
ret;
39
}
40
41 int
main()
42
{
43
44 int k,a,b,c;mem(S,-1);mem(head,-1
);
45 scanf("%d%d%d%d",&n,&m,&p,&
f);
46 go(i,1,n)go(j,1,m)id[i][j]=++
t;
47 go(i,1,f){scanf("%d%d",&a,&b);a=
id[a][b];
48 scanf("%d%d",&b,&c);b=
id[b][c];
49 scanf("%d",&c);S[a][b]=S[b][a]=
c;
50 if(c)ADD(a,b,1<<c),ADD(b,a,1<<
c);}
51
52 go(i,1,n)go(j,1
,m)
53
{
54 a=id[i][j];go(k,0,3
)
55
{
56 int x=i+dx[k],y=j+dy[k];b=
id[x][y];
57 if(S[a][b]==-1&&b)ADD(a,b,1
);
58
}
59
}
60 scanf("%d",&
f);
61 go(i,1,f)scanf("%d%d%d",&a,&b,&c),key[id[a][b]]|=(1<<
c);
62 printf("%d\n"
,spfa());
63 return 0
;
64 }

【大米饼代码】

·最后来一句关于代码疑难的提示:

到达同一个地点时,手上的钥匙越多越好,不要害怕。

比起那些政客的谎言,我们要圣洁得多。

【codevs 1911 孤岛营救问题】的更多相关文章

  1. 孤岛营救问题(BFS+状压DP)

    孤岛营救问题 https://www.luogu.org/problemnew/show/P4011 用状压DP标记拿到钥匙的数量 #include<iostream> #include& ...

  2. loj #6121. 「网络流 24 题」孤岛营救问题

    #6121. 「网络流 24 题」孤岛营救问题   题目描述 1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂, ...

  3. 「LOJ#6121」「网络流 24 题」孤岛营救问题(BFS

    题目描述 1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂,但幸好麦克得到了迷宫的地形图.迷宫的外形是一个长方形 ...

  4. Luogu P4011 孤岛营救问题(状态压缩+最短路)

    P4011 孤岛营救问题 题意 题目描述 \(1944\)年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂,但幸好麦克得到 ...

  5. 【网络流24题】 No.14 孤岛营救问题 (分层图最短路)

    [题意] 1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛, 营救被敌军俘虏的大兵瑞恩. 瑞恩被关押在一个迷宫里, 迷宫地形复杂, 但幸好麦克得到了迷宫的地形图. 迷宫的外形是 ...

  6. [CTSC 1999]拯救大兵瑞恩&[网络流24题]孤岛营救问题

    Description $1944$ 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂,但幸好麦克得到了迷宫的地形图.迷宫 ...

  7. 【刷题】LOJ 6121 「网络流 24 题」孤岛营救问题

    题目描述 1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂,但幸好麦克得到了迷宫的地形图.迷宫的外形是一个长方形 ...

  8. P4011 孤岛营救问题

    \(\color{#0066ff}{题目描述}\) 1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂,但幸好麦克 ...

  9. 洛谷 [P4011] 孤岛营救问题

    状压+BFS 通过观察数据范围可知,我们应该状压钥匙种类,直接BFS即可 注意,一个点处可能不知有一把钥匙 #include <iostream> #include <cstdio& ...

随机推荐

  1. Filter 和 interceptor 的区别

    1. 拦截器 interceptor ● 特点:interceptor 依赖于web框架,在Spring<MV中就是依赖于springMVC框架.在实现上是基于Java的反射机制,属于面向切面编 ...

  2. iOS 11 导航栏 item 偏移问题 和 Swift 下 UIButton 设置 title、image 显示问题

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

  3. nodeJs多进程Cluster

    在前端页面中,如果我们想进行多进程,我们会用到WebWorker,而在NodeJs中,我们如果想充分利用服务器核心资源,我们会用到Node中Cluster模块 直接上代码吧: const cluste ...

  4. 从PRISM开始学WPF(八)導航Navigation?

    0x6Navigation Basic Navigation Prism中的Navigation提供了一种类似导航的功能,他可以根据用户的输入,来刷新UI. 先看一个最简单的例子,通过按钮来导航到一个 ...

  5. AJAX使用说明书

    AJAX简介 什么是AJAX AJAX(Asynchronous Javascript And XML)翻译成中文就是“异步Javascript和XML”.即使用Javascript语言与服务器进行异 ...

  6. Python-进程与线程理论基础-Day10

    进程与线程理论基础 1.背景知识 理论基础: 一 操作系统的作用: 1:隐藏丑陋复杂的硬件接口,提供良好的抽象接口 2:管理.调度进程,并且将多个进程对硬件的竞争变得有序 二 多道技术: 1.产生背景 ...

  7. JVM学习记录

    本博客是为了自己学习JVM而建立,只记录一些自己学习的经过. 最近在看<深入理解Java虚拟机>这本书,里面的内容,很是乏味,因为看不懂所以就会觉得很枯燥,觉得很枯燥看着看着就犯困,然后就 ...

  8. apigw鉴权分析(1-1)阿里数加 - 鉴权方式分析

    一.访问方式 1.访问阿里云首页 https://www.aliyun.com/?utm_medium=text&utm_source=bdbrand&utm_campaign=bdb ...

  9. 翻译:JVM虚拟机规范1.7中的运行时常量池部分(三)

    4.4.7. The CONSTANT_Utf8_info Structure The CONSTANT_Utf8_info structure is used to represent consta ...

  10. "共振式”项目管理

    "共振式”项目管理--是我第一个提出的吗?:) 脑子里突然想到项目管理的一些事情,然后想到项目其实是有节奏的,项目中的人员其实如果找到了这个节奏,踏准了节奏,一切将是顺风顺水. 刚准备动笔时 ...