码迷,mamicode.com
首页 > 其他好文 > 详细

bzoj1001 [BeiJing2006]狼抓兔子

时间:2019-02-03 11:02:37      阅读:107      评论:0      收藏:0      [点我收藏+]

标签:地形   dig   ons   bool   一个   聚集   its   tps   inline   

\(\color{#0066ff}{ 题目描述 }\)

现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的,
而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形:

技术图片

左上角点为(1,1),右下角点为(N,M)(上图中N=4,M=5).有以下三种类型的道路
1:(x,y)<==>(x+1,y)
2:(x,y)<==>(x,y+1)
3:(x,y)<==>(x+1,y+1)
道路上的权值表示这条路上最多能够通过的兔子数,道路是无向的. 左上角和右下角为兔子的两个窝,
开始时所有的兔子都聚集在左上角(1,1)的窝里,现在它们要跑到右下解(N,M)的窝中去,狼王开始伏击
这些兔子.当然为了保险起见,如果一条道路上最多通过的兔子数为K,狼王需要安排同样数量的K只狼,
才能完全封锁这条道路,你需要帮助狼王安排一个伏击方案,使得在将兔子一网打尽的前提下,参与的
狼的数量要最小。因为狼还要去找喜羊羊麻烦.

$\color{#0066ff}{ 输入格式 } $

第一行为N,M.表示网格的大小,N,M均小于等于1000.
接下来分三部分
第一部分共N行,每行M-1个数,表示横向道路的权值.
第二部分共N-1行,每行M个数,表示纵向道路的权值.
第三部分共N-1行,每行M-1个数,表示斜向道路的权值.
输入文件保证不超过10M

\(\color{#0066ff}{输出格式}\)

输出一个整数,表示参与伏击的狼的最小数量.

\(\color{#0066ff}{输入样例}\)

3 4
5 6 4
4 3 1
7 5 3
5 6 7 8
8 7 6 5
5 5 5
6 6 6

\(\color{#0066ff}{输出样例}\)

14

\(\color{#0066ff}{数据范围与提示}\)

none

\(\color{#0066ff}{ 题解 }\)

显然要求一个最小割。。。

于是连上边跑一边最大流最小割即可

连边发现,是双向边,直接连,不用建反向边,因为连的就是双向的

#include<bits/stdc++.h>
#define LL long long
LL in() {
    char ch; LL x = 0, f = 1;
    while(!isdigit(ch = getchar()))(ch == '-') && (f = -f);
    for(x = ch ^ 48; isdigit(ch = getchar()); x = (x << 1) + (x << 3) + (ch ^ 48));
    return x * f;
}
const int maxn = 1001;
struct node {
    int to, dis;
    node *nxt, *rev;
    node(int to = 0, int dis = 0, node *nxt  = NULL): to(to), dis(dis), nxt(nxt) {}
    void *operator new(size_t) {
        static node *S = NULL, *T = NULL;
        return (S == T) && (T = (S = new node[1024]) + 1024), S++;
    }
}*head[maxn * maxn], *cur[maxn * maxn];
int dep[maxn * maxn];
int id[maxn][maxn];
void add(int from, int to, int dis) {
    head[from] = new node(to, dis, head[from]);
}
void link(int from, int to, int dis) {
    add(from, to, dis);
    add(to, from, dis);
    head[from]->rev = head[to];
    head[to]->rev = head[from];
}
int s, t;
bool bfs() {
    for(int i = s; i <= t; i++) cur[i] = head[i], dep[i] = 0;
    std::queue<int> q;
    q.push(s);
    dep[s] = 1;
    while(!q.empty()) {
        int tp = q.front(); q.pop();
        for(node *i = head[tp]; i; i = i->nxt) 
            if(!dep[i->to] && i->dis)
                dep[i->to] = dep[tp] + 1, q.push(i->to);
    }
    return dep[t];
}
int dfs(int x, int change) {
    if(x == t || !change) return change;
    int flow = 0, ls;
    for(node *i = cur[x]; i; i = i->nxt) {
        cur[x] = i;
        if(dep[i->to] == dep[x] + 1 && (ls = dfs(i->to, std::min(change, i->dis)))) {
            flow += ls;
            change -= ls;
            i->dis -= ls;
            i->rev->dis += ls;
            if(!change) break;
        }
    }
    if(!flow) dep[x] = 0;
    return flow;
}
int n, m;
int dinic() {
    int flow = 0;
    while(bfs()) flow += dfs(s, 0x7fffffff);
    return flow;
}
int main() {
    n = in(), m = in();
    for(int i = 1; i <= n; i++)
        for(int j = 1; j <= m; j++) 
            id[i][j] = (i - 1) * m + j;
    s = 1, t = id[n][m];
    int v;
    for(int i = 1; i <= n; i++)
        for(int j = 1; j < m; j++) {
            v = in();
            link(id[i][j], id[i][j + 1], v);
        }
    for(int i = 1; i < n; i++)
        for(int j = 1; j <= m; j++) {
            v = in();
            link(id[i][j], id[i + 1][j], v);
        }
    for(int i = 1; i < n; i++)
        for(int j = 1; j < m; j++) {
            v = in();
            link(id[i][j], id[i + 1][j + 1], v);
        }
    printf("%d\n", dinic());
    return 0;
}

bzoj1001 [BeiJing2006]狼抓兔子

标签:地形   dig   ons   bool   一个   聚集   its   tps   inline   

原文地址:https://www.cnblogs.com/olinr/p/10349581.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!