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

bzoj2127:最小割

时间:2016-01-22 21:56:36      阅读:337      评论:0      收藏:0      [点我收藏+]

标签:

Orz问了师兄好久才懂了,看建图方式就想为什么会对应着三种情况,想流量代表的是什么,后来就画图,果然只有三种情况Orz。。。所以说要多画图多乱搞。。。

s->(i,j),c=w.代表选理科;

(i,j)->t,c=w,代表选文科;

新建k,s->k,c=w,k->(i,j),c=inf,k->(i,j+1),c=inf;代表同时选文或选理;同理。

出现的最小割只会有三种情况,一种是与s相连的都被割掉了,另一种是与t相连的都被割掉了,最后一种是s->k,s->(i,j),k‘->t,(i,j+1)->k‘,被割掉了。分别对应两个人同时选文两个人同时选理和一文一理,然后就OK了。

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

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define rep(i,n) for(int i=1;i<=n;i++)
#define clr(x,c) memset(x,c,sizeof(x))
const int inf=0x3f3f3f3f;
int m,n;
int read(){
   int x=0,f=1;char c=getchar();
   while(!isdigit(c)){
         if(c==‘-‘) f=-1;
         c=getchar();
   }
   while(isdigit(c)){
        x=x*10+c-‘0‘;
        c=getchar();
   }
   return x*f;
}
struct edge{
   int to,cap;
   edge *next,*rev;
};
edge e[300000],*p[50000],*head[50000],*cur[50000],*pt=e;
int d[50000],cnt[50000];
void add(int u,int v,int d){
   pt->to=v;pt->cap=d;pt->next=head[u];head[u]=pt++;
}
void adde(int u,int v,int d){
    add(u,v,d);add(v,u,0);
    head[u]->rev=head[v];
    head[v]->rev=head[u];
}
int maxflow(int s,int t,int n){
    clr(d,0);clr(cnt,0);cnt[0]=n;
    int flow=0,a=inf,x=s;
    while(d[s]<n){
         edge *ee;
         for(ee=cur[x];ee;ee=ee->next)
              if(ee->cap>0&&d[ee->to]+1==d[x]) break;
         if(ee){
               p[ee->to]=cur[x]=ee;
               a=min(a,ee->cap);
               x=ee->to;
               if(x==t){
                      while(x!=s){
                            p[x]->cap-=a;
                            p[x]->rev->cap+=a;
                            x=p[x]->rev->to;
              }
              flow+=a;
              a=inf;
         }
     }else{
         if(!--cnt[d[x]]) break;
         d[x]=n;
         for(ee=head[x];ee;ee=ee->next)
            if(ee->cap>0&&d[ee->to]+1<d[x]){
                 d[x]=d[ee->to]+1;
                 cur[x]=ee;
             }
          cnt[d[x]]++;
          if(x!=s) x=p[x]->rev->to;
     }
   }
    return flow;
}
int get(int x,int y){
     return m*(x-1)+y;
}
int main(){
      n=read(),m=read();
      int s=0,t=n*m+(n-1)*m*2+n*(m-1)*2+1,tot=n*m,ans=0,tmp;
// cout<<t<<" "<<tot<<endl;
       rep(i,n) rep(j,m) adde(s,get(i,j),tmp=read()),ans+=tmp; //这里忘了+=了Orz
       rep(i,n) rep(j,m) adde(get(i,j),t,tmp=read()),ans+=tmp;
       rep(i,n-1) rep(j,m) {
             adde(s,++tot,tmp=read());
             adde(tot,get(i,j),inf);
             adde(tot,get(i+1,j),inf);
             ans+=tmp;
         }
       rep(i,n-1) rep(j,m){
            adde(++tot,t,tmp=read());
            adde(get(i,j),tot,inf);
            adde(get(i+1,j),tot,inf);
            ans+=tmp;
        }
         rep(i,n) rep(j,m-1){
             adde(s,++tot,tmp=read());
            adde(tot,get(i,j),inf);
            adde(tot,get(i,j+1),inf);
            ans+=tmp;
         }
       rep(i,n) rep(j,m-1){
            adde(++tot,t,tmp=read());
            adde(get(i,j),tot,inf);
            adde(get(i,j+1),tot,inf);
            ans+=tmp;
       }
       tot+=2;
   //cout<<tot<<" "<<ans<<endl;
      printf("%d\n",ans-maxflow(s,t,tot));

      return 0;
}

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

2127: happiness

Time Limit: 51 Sec  Memory Limit: 259 MB
Submit: 1255  Solved: 601
[Submit][Status][Discuss]

Description

高一一班的座位表是个n*m的矩阵,经过一个学期的相处,每个同学和前后左右相邻的同学互相成为了好朋友。这学期要分文理科了,每个同学对于选择文科与理科有着自己的喜悦值,而一对好朋友如果能同时选文科或者理科,那么他们又将收获一些喜悦值。作为计算机竞赛教练的scp大老板,想知道如何分配可以使得全班的喜悦值总和最大。

Input

第一行两个正整数n,m。接下来是六个矩阵第一个矩阵为n行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择文科获得的喜悦值。第二个矩阵为n行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择理科获得的喜悦值。第三个矩阵为n-1行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i+1行第j列的同学同时选择文科获得的额外喜悦值。第四个矩阵为n-1行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i+1行第j列的同学同时选择理科获得的额外喜悦值。第五个矩阵为n行m-1列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i行第j+1列的同学同时选择文科获得的额外喜悦值。第六个矩阵为n行m-1列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i行第j+1列的同学同时选择理科获得的额外喜悦值。

Output

输出一个整数,表示喜悦值总和的最大值

Sample Input

1 2
1 1
100 110
1
1000

Sample Output

1210
【样例说明】
两人都选理,则获得100+110+1000的喜悦值。
【数据规模】
对于100%以内的数据,n,m<=100 所有喜悦值均为小于等于5000的非负整数

HINT

 

Source

 
[Submit][Status][Discuss]

 

bzoj2127:最小割

标签:

原文地址:http://www.cnblogs.com/fighting-to-the-end/p/5152157.html

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