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. 2018.3.29 div内容格式设置

    <!DOCTYPE html><html>    <head>        <meta charset="UTF-8">      ...

  2. 网络1711-1712班 c 语言评分总表一览

    学号 姓名 作业地址 PTA实验作业5分 PTA排名2分 阅读代码2分 总结1分 代码规范扣分-2--0 总分 是否推荐博客 1 **莹 http://www.cnblogs.com/wwwwxy12 ...

  3. 支付宝sdk集成,报系统繁忙 请稍后再试(ALI64)

    移动快捷支付,往往需要集成支付宝的sdk,集成的过程相对简单,只要按照支付宝的文档,进行操作一般不会出问题.            下面主要说明一下,集成sdk后报"系统繁忙 请稍后再试(A ...

  4. javascript参数传递中处理+号

    在传值过程中,如果+号也是值的一部分,那就需要对+号进行处理.否则+号会被过滤掉. 处理方式:只需要把js中传过去的+号替换成base64 编码 %2B encodeURI(str).replace( ...

  5. CISCO路由器练习

    前言: 总结了昨天的学习和今天的单臂路由 写了今天的文章. 目录: 路由器的基本配置 单臂路由的练习 正文: 路由器基本配置 环境要求 cisco模拟器 2台交换机 2台PC 1台路由器 路由器介绍: ...

  6. Python 简单聊天室

    #coding=utf-8 from socket import * from threading import Thread import time udpSocket = socket(AF_IN ...

  7. rsync 自动创建目录的坑点

    rsync同步文件有三种模式: 1.把源站路径下某个文件,同步到目标路径.例如rsync -aR /data/1/2/3/a.txt 1.1.1.1:/data/ ,目标机器将自动创建多层目录存放a. ...

  8. EasyUI DataGrid 实现单行/多行编辑功能

    要实现 EasyUI DataGrid 的可编辑很简单,在需要编辑的列添加 editor [编辑器]就可以了. 单行编辑 // 初始化数据列表 function initDatagrid() { $( ...

  9. Android 扩大 View 的点击区域

    有时候,按照视觉图做出来效果后,发现点击区域过小,不好点击,用户体验肯定不好.扩大视图,就会导致整个视觉图变得不好看.那么有没有什么办法在不改变视图大小的前提下扩大点击区域呢? 答案是有! 能够解决这 ...

  10. netty : NioEventLoopGroup 源码分析

    NioEventLoopGroup 源码分析 1. 在阅读源码时做了一定的注释,并且做了一些测试分析源码内的执行流程,由于博客篇幅有限.为了方便 IDE 查看.跟踪.调试 代码,所以在 github ...