POJ 3057 Evacuation(二分匹配)
分析:
这是一个时间和门的二元组(t,d)和人p匹配的问题,当我们固定d0时,(t,d0)匹配的人数和t具有单调性。
t增加看成是多增加了边就行了,所以bfs处理出p到每个d的最短时间,然后把(t,d)和p连边,按t从小到大
枚举点增广就好了。无解的情况只有一种,某个人无论如何都无法出去。
/*********************************************************
* --Sakura hirahira mai orite ochite-- *
* author AbyssalFish *
**********************************************************/
#include<cstdio>
#include<iostream>
#include<string>
#include<cstring>
#include<queue>
#include<vector>
#include<stack>
#include<vector>
#include<map>
#include<set>
#include<algorithm>
#include<cmath>
using namespace std; typedef long long ll; const int XY = , MAX_D = , MAX_P = ;//..MAX_T
char maz[XY][XY+]; const int maxv = MAX_D*MAX_P, maxe = maxv*MAX_P;
int hd[maxv],to[maxe],nx[maxe],ec;
#define eachEage int i = hd[u]; ~i; i = nx[i]
void add(int u,int v)
{
nx[ec] = hd[u];
to[ec] = v;
hd[u] = ec++;
} #define PB push_back
#define resv(x,n) x.reserve(n);
#define PS push
vector<int> pX, pY, dX, dY; const int dx[] = {,,-,}, dy[] = {,-,,}; int dist[XY][XY][XY][XY];
int X,Y; void bfs(int x,int y)
{
int (* const d)[XY] = dist[x][y];
memset(d,0xff,sizeof(dist[x][y])); queue<int> qx, qy;
d[x][y] = ;
qx.PS(x); qy.PS(y);
while(qx.size()){
x = qx.front(); qx.pop();
y = qy.front(); qy.pop();
for(int k = ; k--;){
int nx = x+dx[k], ny = y+dy[k];
if(<=nx && nx<X && <=ny && ny<Y && maz[nx][ny] == '.' && d[nx][ny]<){
d[nx][ny] = d[x][y]+;
qx.PS(nx); qy.PS(ny);
}
}
}
} int link[MAX_P];
int vis[maxv], clk; bool aug(int u)
{
vis[u] = clk;
for(eachEage){
int v = to[i], w = link[v];
if(w< || (vis[w] != clk && aug(w))){
link[v] = u;
return true;
}
}
return false;
} //#define LOCAL
int main()
{
#ifdef LOCAL
freopen("in.txt","r",stdin);
#endif
resv(pX,MAX_P) resv(pX,MAX_P) resv(dX,MAX_D) resv(dY,MAX_D)
int T; cin>>T;
while(T--){
scanf("%d%d",&X,&Y);
pX.clear(); pY.clear();
dX.clear(); dY.clear();
for(int i = ; i < X; i++){
scanf("%s", maz[i]);
for(int j = ; j < Y; j++){
if(maz[i][j] == 'D'){
dX.PB(i); dY.PB(j);
}
else if(maz[i][j] == '.'){
pX.PB(i); pY.PB(j);
}
}
}
int d = dX.size(), p = pX.size();
if(p == ){ puts(""); continue; }
for(int i = ; i < d; i++){
bfs(dX[i],dY[i]);
}
int n = (X-)*(Y-);
bool fail = false;
memset(hd,0xff,sizeof(int)*n*d); ec = ;
for(int i = ; i < p; i++){
bool escape = false;
for(int j = ; j < d; j++){
if(dist[dX[j]][dY[j]][pX[i]][pY[i]] > ){
if(!escape) escape = true;
for(int k = dist[dX[j]][dY[j]][pX[i]][pY[i]]-; k < n; k++){
add(k*d+j, i);
}
}
}
if(!escape){
fail = true; break;
}
}
if(fail){ puts("impossible"); continue; }
memset(link,-,sizeof(int)*p);
int cnt = , ans;
for(int i = ; i < n*d; i++){
clk++;
if(aug(i) && ++cnt == p) {
ans = i/d+;
}
}
printf("%d\n", ans);
}
return ;
}
POJ 3057 Evacuation(二分匹配)的更多相关文章
- POJ 3057 Evacuation (二分匹配)
题意:给定一个图,然后有几个门,每个人要出去,但是每个门每个秒只能出去一个,然后问你最少时间才能全部出去. 析:初一看,应该是像搜索,但是怎么保证每个人出去的时候都不冲突呢,毕竟每个门每次只能出一个人 ...
- TTTTTTTTTTTTT poj 3057 Evacuation 二分图匹配+bfs
题意:见挑战230页 #include <iostream> #include <cstdio> #include <cstring> #include <c ...
- POJ 3057 Evacuation 二分+最大流
Evacuation 题目连接: http://poj.org/problem?id=3057 Description Fires can be disastrous, especially when ...
- POJ 3057 Evacuation 二分图匹配
每个门每个时间只能出一个人,那就把每个门拆成多个,对应每个时间. 不断增加时间,然后增广,直到最大匹配. //#pragma comment(linker, "/STACK:10240000 ...
- POJ 3057 Evacuation 题解
题目 Fires can be disastrous, especially when a fire breaks out in a room that is completely filled wi ...
- POJ 3041 - 最大二分匹配
这道题实现起来还是比较简单的,但是理解起来可能有点困难. 我最开始想到的是贪心法,每次消灭当前小行星最多的一行或一列.然而WA了.Discuss区里已经有高人给出反例. 下面给出正确的解法 我们把行和 ...
- POJ 3057 Evacuation(二分图匹配+BFS)
[题目链接] http://poj.org/problem?id=3057 [题目大意] 给出一个迷宫,D表示门,.表示人,X表示不可通行, 每个门每时间单位只允许一个人通过, 每个人移动一格的为一时 ...
- 【最大匹配+二分答案】POJ 3057 Evacuation
题目大意 POJ链接 有一个\(X×Y\)的房间,X代表墙壁,D是门,.代表人.这个房间着火了,人要跑出去,但是每一个时间点只有一个人可以从门出去. 问最后一个人逃出去的最短时间,如果不能逃出去,输出 ...
- poj 2446 Chessboard (二分匹配)
Chessboard Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 12800 Accepted: 4000 Descr ...
随机推荐
- VS Code(待补充)
使用! 然后Tab 快速生成html文档结构 快速生成一个类 .类名 会有提示 .container.box .col-6*2 VisualStudio Code怎么同时编辑多处?
- MongoDB的安装避坑(踩坑)
下载 可以去官网下载:https://www.mongodb.com/download-center/community 安装 下载完了就可以使用安装包安装:我下载的mongodb版本是:v4.0.9 ...
- LibreOJ #6192. 「美团 CodeM 复赛」城市网络
#6192. 「美团 CodeM 复赛」城市网络 内存限制:64 MiB时间限制:500 ms标准输入输出 题目类型:传统评测方式:文本比较 上传者: sqc 提交提交记录统计讨论测试数据 题目描 ...
- 2014-11-3 NOIP模拟赛2
NOIP 2014 水题模拟赛 (请选手务必仔细阅读本页内容) 一.题目概况 中文题目名称 数列 刷漆 排队 英文题目与子目录名 seq paint layout 可执行文件名 seq paint l ...
- [题解](排列/逆序对)luogu_P1338末日的传说
首先我们要考虑怎么排能使逆序对数最多:显然是下降序列时,会产生n*(n-1)/2数量的逆序对 那么我们肯定是要尽量把序列的尾端安排成下降序列,前面的尽量不动,中间可能有一段排列自适应到m的逆序对数 然 ...
- 瓷砖覆盖(状压DP)
题目描述 Description 用1*2的瓷砖去铺N*M的地面,问有多少种铺法 输入描述 Input Description 第一行有两数n,m.表示地面的大小 输出描述 Output Descri ...
- 【ACM】取石子 - 博弈论
取石子(一) 时间限制:3000 ms | 内存限制:65535 KB 难度:2 描述 一天,TT在寝室闲着无聊,和同寝的人玩起了取石子游戏,而由于条件有限,他/她们是用旺仔小馒头当作石子.游 ...
- 剑指Offer——数组中的逆序对(归并排序的应用)
蛮力: 遍历数组,对每个元素都往前遍历所有元素,如果有发现比它小的元素,就count++. 最后返回count取模. 结果没问题,但超时哈哈哈,只能过50%. 归并法: 看讨论,知道了这道题的经典 ...
- UiAutomator环境配置
前提:下载jdk以及adt 一.配置java开发环境 1.新建系统变量 -——变量名:JAVA_HOME 变量值:C:\Program Files\Java\jdk1.7.0_80 2.编辑 ...
- Java面向对象_适配器模式
概念:将一个类的接口转换成客户希望的另外一个接口.适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作 public class Practice14 { public static v ...