Description

$1944$ 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩。瑞恩被关押在一个迷宫里,迷宫地形复杂,但幸好麦克得到了迷宫的地形图。迷宫的外形是一个长方形,其南北方向被划分为 $N$ 行,东西方向被划分为 $M$ 列,于是整个迷宫被划分为 $N\times M$ 个单元。每一个单元的位置可用一个有序数对(单元的行号,单元的列号)来表示。南北或东西方向相邻的 $2$ 个单元之间可能互通,也可能有一扇锁着的门,或者是一堵不可逾越的墙。迷宫中有一些单元存放着钥匙,并且所有的门被分成$P$ 类,打开同一类的门的钥匙相同,不同类门的钥匙不同。

大兵瑞恩被关押在迷宫的东南角,即 $(N,M)$ 单元里,并已经昏迷。迷宫只有一个入口,在西北角。也就是说,麦克可以直接进入 $(1,1)$ 单元。另外,麦克从一个单元移动到另一个相邻单元的时间为 $1$,拿取所在单元的钥匙的时间以及用钥匙开门的时间可忽略不计。

试设计一个算法,帮助麦克以最快的方式到达瑞恩所在单元,营救大兵瑞恩。

Input

第 $1$ 行有 $3$ 个整数,分别表示 $N,M,P$ 的值。

第 $2$ 行是 $1$ 个整数 $K$,表示迷宫中门和墙的总数。

第 $I+2$ 行$(1\leq I\leq K)$,有 $5$ 个整数,依次为$X_{i1},Y_{i1},X_{i2},Y_{i2},G_i$:

  • 当 $G_i \geq 1$ 时,表示 $(X_{i1},Y_{i1})$ 单元与 $(X_{i2},Y_{i2})$ 单元之间有一扇第 $G_i$ 类的门

  • 当 $G_i=0$ 时,表示 $(X_{i1},Y_{i1})$ 单元与 $(X_{i2},Y_{i2})$ 单元之间有一堵不可逾越的墙(其中,$|X_{i1}-X_{i2}|+|Y_{i1}-Y_{i2}|=1$,$0\leq G_i\leq P$)。

第 $K+3$ 行是一个整数 $S$,表示迷宫中存放的钥匙总数。

第 $K+3+J$ 行$(1\leq J\leq S)$,有 $3$ 个整数,依次为 $X_{i1},Y_{i1},Q_i$:表示第 $J$ 把钥匙存放在 $(X_{i1},Y_{i1})$单元里,并且第 $J$ 把钥匙是用来开启第 $Q_i$ 类门的。(其中$1\leq Q_i\leq P$)。

输入数据中同一行各相邻整数之间用一个空格分隔。

Output

将麦克营救到大兵瑞恩的最短时间的值输出。如果问题无解,则输出 $-1$。

Sample Input

4 4 9
9
1 2 1 3 2
1 2 2 2 0
2 1 2 2 0
2 1 3 1 0
2 3 3 3 0
2 4 3 4 1
3 2 3 3 0
3 3 4 3 0
4 3 4 4 0
2
2 1 2
4 2 1

Sample Output

14

HINT

$|X_{i1}-X_{i2}|+|Y_{i1}-Y_{i2}|=1,0\leq G_i\leq P$

$1\leq Q_i\leq P$

$N,M,P\leq10, K<150,S\leq 14$

题解

比较经典的分层图。

状压钥匙状态,按状态分层,跑最短路。

(代码好早之前写的,风格早就不一样了)

 #include<cmath>
#include<ctime>
#include<queue>
#include<stack>
#include<cstdio>
#include<string>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int w1[]={-,,,};
const int w2[]={,-,,}; int n,m,p,k,X1,X2,Y1,Y2,c,P;
int key[][][][]; struct tt
{
int to,next,cost;
}edge[];
int path[],top;
int dist[]; void Add(int x,int y,int c);
int Bfs(); int main()
{
scanf("%d%d%d%d",&n,&m,&p,&k);
P=<<p;
for (int i=;i<=k;i++)
{
scanf("%d%d%d%d%d",&X1,&Y1,&X2,&Y2,&c);
if (c==) c=-;
key[X1][Y1][X2][Y2]=key[X2][Y2][X1][Y1]=c;
}
for (int i=;i<P;i++)
{
for (int x=;x<=n;x++)
{
for (int y=;y<=m;y++)
{
for (int w=;w<;w++) if (x+w1[w]>=&&x+w1[w]<=n&&y+w2[w]>=&&y+w2[w]<=m)
{
if (key[x][y][x+w1[w]][y+w2[w]]==-) continue;
if (key[x][y][x+w1[w]][y+w2[w]]==) Add(x*m*P+y*P+i,(x+w1[w])*m*P+(y+w2[w])*P+i,);
else if (i&(<<(key[x][y][x+w1[w]][y+w2[w]]-))) Add(x*m*P+y*P+i,(x+w1[w])*m*P+(y+w2[w])*P+i,);
}
}
}
}
scanf("%d",&k);
for (int i=;i<=k;i++)
{
scanf("%d%d%d",&X1,&Y1,&c);
for (int j=;j<P;j++) if (!(j&(<<(c-))))
{
Add(X1*m*P+Y1*P+j,X1*m*P+Y1*P+(j|(<<(c-))),);
}
}
printf("%d\n",Bfs());
return ;
} void Add(int x,int y,int c)
{
edge[++top].to=y;
edge[top].next=path[x];
edge[top].cost=c;
path[x]=top;
}
int Bfs()
{
memset(dist,/,sizeof(dist));
int INF=dist[m*P+P];
dist[m*P+P]=;
queue<int>Q;
Q.push(m*P+P);
while (!Q.empty())
{
for (int i=path[Q.front()];i;i=edge[i].next)
{
if (dist[Q.front()]+edge[i].cost<dist[edge[i].to])
{
dist[edge[i].to]=dist[Q.front()]+edge[i].cost;
Q.push(edge[i].to);
}
}
Q.pop();
}
int ans=INF;
for (int i=;i<P;i++) if (dist[n*m*P+m*P+i]<ans) ans=dist[n*m*P+m*P+i];
return ans==INF ? -:ans;
}

[CTSC 1999]拯救大兵瑞恩&[网络流24题]孤岛营救问题的更多相关文章

  1. CTSC 1999 家园 【网络流24题】星际转移

    直接把每一个点,每一天拆成一个点. 然后每个点到下一天连$inf$的边. 然后把飞船的路径用容量为飞船容量的边连接. 然后跑网络流判断是否满流. #include <queue> #inc ...

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

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

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

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

  4. 【线性规划与网络流 24题】已完成(3道题因为某些奇怪的原因被抛弃了QAQ)

    写在前面:SDOI2016 Round1滚粗后蒟蒻开始做网络流来自我拯救(2016-04-11再过几天就要考先修课,现在做网络流24题貌似没什么用←退役节奏) 做的题目将附上日期,见证我龟速刷题. 1 ...

  5. 【算法】【网络流24题】巨坑待填(成功TJ,有时间再填)

    ------------------------------------------------------------------------------------ 17/24 --------- ...

  6. 网络流基础&网络流24题

    网络最大流 dinic+当前弧优化. const int N=10007,M=100007,inf=1e9; int s,t,head[N],ver[M],edge[M],Next[M],tot=1, ...

  7. Cogs 728. [网络流24题] 最小路径覆盖问题

    [网络流24题] 最小路径覆盖问题 ★★☆ 输入文件:path3.in 输出文件:path3.out 评测插件 时间限制:1 s 内存限制:128 MB 算法实现题8-3 最小路径覆盖问题(习题8-1 ...

  8. COGS743. [网络流24题] 最长k可重区间集

    743. [网络流24题] 最长k可重区间集 ★★★   输入文件:interv.in   输出文件:interv.out   简单对比时间限制:1 s   内存限制:128 MB «问题描述: «编 ...

  9. Cogs 14. [网络流24题] 搭配飞行员

    这道题其实蛮好想的,因为分为正,副飞行员.所以就把正飞行员当作Boy,副飞行员当作Girl.然后做Hungry即可. #include<bits/stdc++.h> using names ...

随机推荐

  1. 201621123062《java程序设计》第13周作业总结

    1. 本周学习总结 以你喜欢的方式(思维导图.OneNote或其他)归纳总结多网络相关内容. 思维导图: 2. 为你的系统增加网络功能(购物车.图书馆管理.斗地主等)-分组完成 为了让你的系统可以被多 ...

  2. 201621123043《java程序设计》第五周学习总结

    1. 本周学习总结 1.1 写出你认为本周学习中比较重要的知识点关键词 接口. Comparable接口 .Comparator接口.compareTo. 1.2 尝试使用思维导图将这些关键词组织起来 ...

  3. 咬碎STL空间配置器

    STL空间配置器 一.开场白: 给我的感觉就是,了解是空间配置器的功能,是那么的明了:在看原理,我还是很开心:接下来是360度大转变: 那么长的变量或者函数命名.那么多的宏.不爽,不过,遇上我这种二货 ...

  4. (转)如何在Eclipse中查看JDK类库的源代码

    在Eclipse中查看JDK类库的源代码!!! 设置: 1.点 “window”-> "Preferences" -> "Java" -> & ...

  5. 【Swift】Runtime动态性分析

    Swift是苹果2014年发布的编程开发语言,可与Objective-C共同运行于Mac OS和iOS平台,用于搭建基于苹果平台的应用程序.Swift已经开源,目前最新版本为2.2.我们知道Objec ...

  6. python的Flask 介绍

    Flask 介绍 知识点 微框架.WSGI.模板引擎概念 使用 Flask 做 web 应用 模板的使用 根据 URL 返回特定网页 实验步骤 1. 什么是 Flask? Flask 是一个 web ...

  7. TCP/IP协议复习

  8. PostgreSQL 配置安装

    Mac 安装 http://postgresapp.com/ 创建和删除数据库用户 对应命令如下(在postgres=# 环境下):1.查看数据库用户列表: \du2.创建数据库用户: create ...

  9. es6对象字面量增强

    相对于ES5,ES6的对象字面量得到了很大程度的增强.这些改进我们可以输入更少的代码同时语法更易于理解.那就一起来看看对象增强的功能.对象字面量简写(Object Literal Shorthand) ...

  10. 偶遇vue-awesome-swiper的坑

    最近用vue重构一个移动端的项目,碰到了不少坑,今天拿移动端最著名的轮播插件swiper为例来说,由于这个项目没用UI库,纯手写的样式,沿用老的插件,自然而然的选择了vue-awesome-swipe ...