我们可以BFS出每个出口到每个人的最短距离, 然后二分答案, 假设当前答案为m, 把一个出口拆成m个表示m个时间, 点u到出口v的距离为d, 那么u->v的[d, m]所有点连边, 然后跑匈牙利去check就行了...其实这道题挺好想但是码量还是挺大的....

-----------------------------------------------------------------------------

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
 
using namespace std;
 
#define chk(r, c) (r >= 0 && r < N && c >= 0 && c < M)
 
const int maxn = 29;
const int MAXN = 330;
const int MAX_V = 80;
const int INF = 0X3F3F3F3F;
const int dir[4][2] = {{0, -1}, {0, 1}, {-1, 0}, {1, 0}};
 
struct edge {
int to;
edge* next;
} E[8000000], *pt, *head[MAXN];
 
void InitEdge() {
pt = E;
memset(head, 0, sizeof head);
}
 
void AddEdge(int u, int v) {
pt->to = v; pt->next = head[u]; head[u] = pt++;
}
 
int D[MAX_V][MAXN];
int R[MAXN], C[MAXN], _R[MAX_V], _C[MAX_V], Id[maxn][maxn];
int vis[MAXN * MAXN], mch[MAXN * MAXN];
int CK, N, M, V, _V;
char s[maxn][maxn];
queue<int> Q;
 
bool Dfs(int x) {
for(edge* e = head[x]; e; e = e->next) if(vis[e->to] != CK) {
vis[e->to] = CK;
if(!~mch[e->to] || Dfs(mch[e->to])) {
mch[e->to] = x;
return true;
}
}
return false;
}
 
void BFS(int x) {
memset(D[x], INF, sizeof D[x]);
while(!Q.empty()) Q.pop();
int X = _R[x], Y = _C[x];
if((X == N - 1 && Y == M - 1) || (!X && !Y)) return;
if(!X && ~Id[X + 1][Y]) {
Q.push(Id[X + 1][Y]);
D[x][Id[X + 1][Y]] = 0;
}
if(X == N - 1 && ~Id[X - 1][Y]) {
Q.push(Id[X - 1][Y]);
D[x][Id[X - 1][Y]] = 0;
}
if(!Y&& ~Id[X][Y + 1]) {
Q.push(Id[X][Y + 1]);
D[x][Id[X][Y + 1]] = 0;
}
if(Y == M - 1 && ~Id[X][Y - 1]) {
Q.push(Id[X][Y - 1]);
D[x][Id[X][Y - 1]] = 0;
}
while(!Q.empty()) {
int v = Q.front(); Q.pop();
int r = R[v], c = C[v];
for(int i = 0; i < 4; i++) {
int _r = r + dir[i][0], _c = c + dir[i][1];
if(!chk(_r, _c) || s[_r][_c] != '.') continue;
int _v = Id[_r][_c];
if(~_v && D[x][v] + 1 < D[x][_v]) {
D[x][_v] = D[x][v] + 1;
Q.push(_v);
}
}
}
}
 
void Init() {
memset(Id, -1, sizeof Id);
V = _V = 0;
scanf("%d%d", &N, &M);
for(int i = 0; i < N; i++) {
scanf("%s", s[i]);
for(int j = 0; j < M; j++) switch(s[i][j]) {
case '.' : R[V] = i; C[V] = j; Id[i][j] = V++; break;
case 'D' : _R[_V] = i; _C[_V] = j; Id[i][j] = _V++; break;
default : break;
}
}
for(int i = 0; i < _V; i++) BFS(i);
}
 
void InitArray() {
memset(vis, -1, sizeof vis);
memset(mch, -1, sizeof mch);
}
 
bool Jud(int m) {
InitEdge();
InitArray();
for(int i = 0; i < _V; i++) {
int B = i * m;
for(int j = 0; j < V; j++)
for(int k = D[i][j]; k < m; k++)
AddEdge(j, B + k);
}
int cnt = 0;
for(CK = 0; CK < V; CK++)
if(Dfs(CK)) cnt++;
return cnt == V;
}
 
bool Imp() {
memset(vis, 0, sizeof vis);
for(int i = 0; i < _V; i++)
for(int j = 0; j < V; j++)
if(D[i][j] != INF) vis[j] = -1;
for(int i = 0; i < V; i++)
if(~vis[i]) return false;
return true;
}
 
int main() {
Init();
if(!Imp()) {
puts("impossible"); return 0;
}
int l = 1, r = V, ans = 0;
while(l <= r) {
int m = (l + r) >> 1;
if(Jud(m))
ans = m, r = m - 1;
else
l = m + 1;
}
printf("%d\n", ans);
return 0;
}

-----------------------------------------------------------------------------

1189: [HNOI2007]紧急疏散evacuate

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 1219  Solved: 448
[Submit][Status][Discuss]

Description

发生了火警,所有人员需要紧急疏散!假设每个房间是一个N M的矩形区域。每个格子如果是'.',那么表示这是一块空地;如果是'X',那么表示这是一面墙,如果是'D',那么表示这是一扇门,人们可以从这儿撤出房间。已知门一定在房间的边界上,并且边界上不会有空地。最初,每块空地上都有一个人,在疏散的时候,每一秒钟每个人都可以向上下左右四个方向移动一格,当然他也可以站着不动。疏散开始后,每块空地上就没有人数限制了(也就是说每块空地可以同时站无数个人)。但是,由于门很窄,每一秒钟只能有一个人移动到门的位置,一旦移动到门的位置,就表示他已经安全撤离了。现在的问题是:如果希望所有的人安全撤离,最短需要多少时间?或者告知根本不可能。

Input

输入文件第一行是由空格隔开的一对正整数N与M,3<=N <=20,3<=M<=20,以下N行M列描述一个N M的矩阵。其中的元素可为字符'.'、'X'和'D',且字符间无空格。

Output

只有一个整数K,表示让所有人安全撤离的最短时间,如果不可能撤离,那么输出'impossible'(不包括引号)。

Sample Input

5 5
XXXXX
X...D
XX.XX
X..XX
XXDXX

Sample Output

3

HINT

Source

BZOJ 1189: [HNOI2007]紧急疏散evacuate( BFS + 二分答案 + 匈牙利 )的更多相关文章

  1. bzoj 1189 [HNOI2007]紧急疏散evacuate 二分+网络流

    [HNOI2007]紧急疏散evacuate Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3626  Solved: 1059[Submit][St ...

  2. 【BZOJ】1189: [HNOI2007]紧急疏散evacuate(二分+bfs+网络流)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1189 表示完全不会QAQ.... 于是膜拜题解orz 二分时间........... 于是转换成判定 ...

  3. bzoj 1189: [HNOI2007]紧急疏散evacuate 分层图最大流_拆点_二分

    Description 发生了火警,所有人员需要紧急疏散!假设每个房间是一个N M的矩形区域.每个格子如果是'.',那么表示这是一 块空地:如果是'X',那么表示这是一面墙,如果是'D',那么表示这是 ...

  4. BZOJ 1189 [HNOI2007]紧急疏散evacuate

    Description 发生了火警,所有人员需要紧急疏散!假设每个房间是一个N M的矩形区域.每个格子如果是'.',那么表示这是一块空地:如果是'X',那么表示这是一面墙,如果是'D',那么表示这是一 ...

  5. 【洛谷 P3191】 [HNOI2007]紧急疏散EVACUATE(二分答案,最大流)

    题目链接 sb错误调了3hour+.. bfs预处理出每个\(.\)到每个\(D\)的最短距离. 二分时间\(t\),把每个\(D\)拆成\(t\)个点,这\(t\)个点两两连边,流量\(INF\)表 ...

  6. 1189: [HNOI2007]紧急疏散evacuate - BZOJ

    Description 发生了火警,所有人员需要紧急疏散!假设每个房间是一个N M的矩形区域.每个格子如果是'.',那么表示这是一块空地:如果是'X',那么表示这是一面墙,如果是'D',那么表示这是一 ...

  7. 【bzoj1189】[HNOI2007]紧急疏散evacuate BFS最短路+动态加边网络流

    题目描述 发生了火警,所有人员需要紧急疏散!假设每个房间是一个N M的矩形区域.每个格子如果是'.',那么表示这是一块空地:如果是'X',那么表示这是一面墙,如果是'D',那么表示这是一扇门,人们可以 ...

  8. 1189: [HNOI2007]紧急疏散evacuate

    Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3831  Solved: 1119[Submit][Status][Discuss] Descript ...

  9. BZOJ1189 [HNOI2007]紧急疏散evacuate 【二分 + 网络流】

    题目 发生了火警,所有人员需要紧急疏散!假设每个房间是一个N M的矩形区域.每个格子如果是'.',那么表示这是一 块空地:如果是'X',那么表示这是一面墙,如果是'D',那么表示这是一扇门,人们可以从 ...

随机推荐

  1. poj2070简单题

    #include <stdio.h> #include <stdlib.h> int main() { float sped; int wei,sth; while(scanf ...

  2. mongodb remove删除文档的用法

    在看<mongoDB权威指南>中,在删除文档时,出现问题: 书中介绍:采用db.foo.remove()命令则可以删除foo集合中所有的文档,但是在执行该命令时,shell客户端却报错. ...

  3. 【菜鸟学习Linux】-第三章- Linux环境搭建-使用VMware9安装Ubuntu 12.04系统

    上一步,我们安装了VMware9虚拟机,现在我们就是用它来安装Ubuntu12.04系统,至于Ubuntu是什么,我就不废话了,大家google一下,比我讲的清楚,好了,开始干活! Ubuntu官网下 ...

  4. 前端新人学习笔记-------html/css/js基础知识点

    即将毕业的软件工程大学生一枚,秋季招聘应聘的是Android,今年来到公司实习,要求做前端开发,所以一切只有现学,现在根据视频来学习,然后开这个博客记录一下自己的学习过程,废话不多说,开写. 4月6日 ...

  5. HTML系列(四):编辑图像

    一.图像的基本概念 1.矢量图:文件占用空间小,放大后图像不会失真,和分辨率无关.适用于图形设计.文字设计.标志设计.版式设计等. 2.位图:由像素点组成,文件较大,放大和缩小图像会失真. 3.有损压 ...

  6. ASPxGridView-为每行添加序号

    添加一个新的非绑定列,使用CustomColumnDisplayText事件来分配序号给该列 <dx:GridViewDataTextColumn Caption="序号" ...

  7. GWT工程 —— HostedMode(宿主模式下调试) 所有的运行命令

    Unknown argument: -helpGoogle Web Toolkit 1.7.0HostedMode [-noserver] [-port port-number | "aut ...

  8. c++中派生类对基类成员的三种访问规则(转)

    C++中派生类对基类成员的访问形式主要有以下两种:1.内部访问:由派生类中新增成员对基类继承来的成员的访问.2.对象访问:在派生类外部,通过派生类的对象对从基类继承来的成员的访问.今天给大家介绍在3中 ...

  9. C - 下沙小面的(2)

    C – 下沙小面的(2)Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64uSubmit StatusDes ...

  10. T-SQL应用,视图、存储过程、触发器、游标、临时表等

    sqlserver常用操作: 视图.存储过程.触发器.函数 --*********************批处理********************* --[在一个批处理中存有一个语法错误,则所有 ...