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. 测试工作中经常用到的几个Linux命令(第一弹)

    自己平时测试工作中经常要在Linux下搭建测试环境,有涉及到启动/终止服务器,修改tomcat配置文件,偶尔碰到端口被占用... 这时就不得不需要一些基本的Linux命令来处理遇到的这些问题(顺便迈向 ...

  2. 集大1513 & 1514班 软件工程第二次作业评分与点评

    谢谢按时完成作业的同学. 请大家在今后的作业中多思考,认真完成并注意作业的原创性. 学号 作业标题 作业地址 提交日期 分数 201521121087 微信APP简要分析 http://www.cnb ...

  3. 个人技术博客(alpha)

    APP的权限校验不同于web网页端,web一般使用session记录用户的状态信息,而app则使用token令牌来记录用户信息.有这样一个场景,系统的数据量达到千万级,需要几台服务器部署,当一个用户在 ...

  4. ajax实现无刷新分页效果

    基于jquery.pagination.js实现的无刷新加载分页数据效果. 简介与说明 * 该插件为Ajax分页插件,一次性加载数据,故分页切换时无刷新与延迟.如果数据量较大,加载会比较慢. * 分页 ...

  5. javascript实现小鸟飞行轨迹

    javascript实现小鸟飞行轨迹 代码如下:

  6. 读取.properties的内容1

    属性文件方便于项目进行更改,在项目开发中用的也是非常的普遍,在这里就把属性文件的读取通过代码进行一个小结: package com.oyy.test; import java.io.BufferedI ...

  7. 调用WCF时,调用已超过传入消息(65536)的最大消息大小配额。若要增加配额,请使用相应绑定。

    解决方案: 其实只要在客户端配置文件中加上如下紫色粗体属性( maxReceivedMessageSize): <?xml version="1.0" encoding=&q ...

  8. JS 上传图片时实现预览

    网页中一张图片可以这样显示: <img src="http://www.letuknowit.com/images/wg.png"/>也可以这样显示:<img s ...

  9. Python内置函数(28)——iter

    英文文档: iter(object[, sentinel]) Return an iterator object. The first argument is interpreted very dif ...

  10. python 单例模式的四种创建方式

    单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场. ...