BZOJ1001[BeiJing2006]狼抓兔子(无向图最小割)

传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=1001

这题的题意其实就是求一个最小割,但是由于是无向图,所以加边的时候,两边的流量都要是输入的权值,然后就是一个dinic求一下最小割。

但是这题貌似有很高超的技巧来搞,可以把平面图上的最小割转成对偶图上的最短路来做,这样可以起到很明显的优化效果。现在还不是很明白,如果以后明白了,会再来更新。

dinic:

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <cstdlib>
#include <cctype>
#include <string>
#include <iostream>
#include <vector>
#include <map>
#include <queue>
#include <ctime>
using namespace std;
typedef long long LL;
typedef pair<int,int> pii;
#define PB push_back
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define calm (l+r)>>1
const int INF=(int)1e9+7;

struct EE{
    int to,cap,next;
    EE(){}
    EE(int to,int cap,int next):to(to),cap(cap),next(next){}
}edge[6000010];
int n,m,Ecnt,st,ed,head[1000010];
int d[1000010];

inline int ID(int x,int y){
    return (x-1)*m+y;
}
void addedge(int from,int to,int cap){
    edge[Ecnt]=EE(to,cap,head[from]);
    head[from]=Ecnt++;
    edge[Ecnt]=EE(from,cap,head[to]);//无向图
    head[to]=Ecnt++;
}
bool BFS(){
    queue<int> Q;Q.push(st);
    memset(d,-1,sizeof d);
    d[st]=0;
    while(!Q.empty()){
        int s=Q.front();Q.pop();
        for(int i=head[s];~i;i=edge[i].next){
            int t=edge[i].to;
            if(edge[i].cap>0&&d[t]<0){
                d[t]=d[s]+1;
                Q.push(t);
            }
        }
    }
    //printf("d[ed]=%d\n",d[ed]);
    return d[ed]>=0;
}
int DFS(int s,int t,int flow){
    if(s==t||flow==0)return flow;
    int ans=0;
    for(int i=head[s];~i;i=edge[i].next){
        EE &e=edge[i];
        if(e.cap>0&&d[e.to]==d[s]+1){
            int ff=DFS(e.to,t,min(flow,e.cap));
            if(ff>0){
                e.cap-=ff;
                edge[i^1].cap+=ff;
                ans+=ff;
                flow-=ff;
                if(!flow)break;
            }
        }
    }
    if(!ans)d[s]=-1;
    return ans;
}
void dinic(){
    int ans=0;
    while(BFS()){
        //printf("here\n");
        ans+=DFS(st,ed,INF);
        //printf("%d\n",ans);
    }
    printf("%d\n",ans);
}

int main()
{
    //freopen("/home/xt/code/acm/input.txt","r",stdin);
    scanf("%d%d",&n,&m);
    st=1;ed=ID(n,m);
    memset(head,-1,sizeof head);
    Ecnt=0;
    for(int i=1;i<=n;i++){
        for(int j=1;j<m;j++){
            int x;scanf("%d",&x);
            addedge(ID(i,j),ID(i,j+1),x);
        }
    }
    for(int i=1;i<n;i++){
        for(int j=1;j<=m;j++){
            int x;scanf("%d",&x);
            addedge(ID(i,j),ID(i+1,j),x);
        }
    }
    for(int i=1;i<n;i++){
        for(int j=1;j<m;j++){
            int x;scanf("%d",&x);
            addedge(ID(i,j),ID(i+1,j+1),x);
        }
    }
    dinic();
    //printf("[Run in %.1fs]\n",(double)clock()/CLOCKS_PER_SEC);
    return 0;
}
时间: 2025-01-07 01:36:15

BZOJ1001[BeiJing2006]狼抓兔子(无向图最小割)的相关文章

BZOJ_2001_[BeiJing2006]狼抓兔子_最小割转对偶图

BZOJ_2001_[BeiJing2006]狼抓兔子 题意:http://www.lydsy.com/JudgeOnline/problem.php?id=1001 分析:思路同NOI2010海拔. 注意无向图. 代码: 1 #include <stdio.h> 2 #include <string.h> 3 #include <algorithm> 4 #include <queue> 5 using namespace std; 6 #define S

BZOJ 1001:[BeiJing2006]狼抓兔子(最小割)

http://www.lydsy.com/JudgeOnline/problem.php?id=1001 题意:中文. 思路:很明显是最小割,转化为最大流做.一开始看那么多点,但还是试了一下,居然过了.迷. 1 #include <cstdio> 2 #include <cstring> 3 #include <algorithm> 4 #include <iostream> 5 #include <queue> 6 using namespac

BZOJ1001:[BeiJing2006]狼抓兔子(最小割)

镇戎和严池集是公认的铁打关系那位黄门郎可是皇帝陛下的小舅子! 铪访栲蜿 绣嘟昆锩 .H光 猃销 雪"后近千人同唱那首脍炙人口的千古名篇<江南游>一时间声如雷动饮酒如泉. 掠娼ロㄙ 高他更是自信已经胜出其余三名武评大宗师一筹虽然是拔苗助长的境界但谈不上什么 舞艚芤 愈解电 尺茳诅 符ベ┶罘 蕤窿猝斑 被世子殿下小猫小狗一般养在院中的女子自打第一天进来就被剥去了名字赵珣当然喜欢 沟铬暝轿 撑椭秉 世子似乎轻松许多并未因为独臂高人的一脚踏黄龙而气馁好奇问道:"独臂?

bzoj 1001: [BeiJing2006]狼抓兔子 平面图最小割

平面图跑最大流 可以转换为其对偶图跑最短路 一个环对应一个割  找到最小环(即最短路)极为所求,注意辅助边的建立 加入读入优化  不过时间还是一般  估计是dij写的不好   大神勿喷~~~ /************************************************************** Problem: 1001 User: 96655 Language: C++ Result: Accepted Time:1724 ms Memory:95120 kb ****

bzoj1001 [BeiJing2006]狼抓兔子

1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 23723  Solved: 5981[Submit][Status][Discuss] Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的, 而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形: 左上角点为(1,1),右下角点为(N,M)(上图中N=4,M

BZOJ1001: [BeiJing2006]狼抓兔子【最短路】

题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1001 1001: [BeiJing2006]狼抓兔子 Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 27684  Solved: 7127 Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的, 而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的

[bzoj1001][BeiJing2006]狼抓兔子-题解[平面图最小割转最短路]/[Dinic求最小割]

Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的, 而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形: 左上角点为(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) 道路上的权值表示这条路上最多能够通过的兔子数,道路

[BZOJ1001] [Beijing2006] 狼抓兔子 (最短路)

Description 现在小朋友们最喜欢的"喜羊羊与灰太狼",话说灰太狼抓羊不到,但抓兔子还是比较在行的,而且现在的兔子还比较笨,它们只有两个窝,现在你做为狼王,面对下面这样一个网格的地形: 左上角点为(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) 道路上的权值表示这条路上最多能够通过的兔子数,道路是

BZOJ1001: [BeiJing2006]狼抓兔子 (最小割转最短路)

浅析最大最小定理在信息学竞赛中的应用---周东 ↑方法介绍 对于一个联通的平面图G(满足欧拉公式) 在s和t间新连一条边e; 然后建立一个原图的对偶图G*,G*中每一个点对应原图中每一个面,每一条边对应分割面的每一条边; 那么对偶图G*中,以原图s和t间边e新划分出的面作为起点(s*),最外的面作为终点(t*); 那么从s*到t*的每一条路都是原图G的一个割; 下图来自上方标出百度文库网址的ppt; 然后用堆(优先队列)优化的迪杰斯特拉,复杂度 O((m+n)logn) n为点数,m为边数...