HDU 3277 Marriage Match III(中等,好题)[二分最大流]+拆点

题意:基本与3081一样,但是多了每个女孩可以另外选任意k个不喜欢的
思路:拆点。把点i拆分为i和i+n,i到i+n连一条权为k的边,i与j连一条权是1的边。

#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring>
using namespace std;
#define cl(a,b) memset(a,b,sizeof(a))
const int maxn=805;
const int inf=1<<28;

template<int nv,int ne>
struct isap{
    int n,size;
    int head[nv];
    int dis[nv],gap[nv],cur[nv],pre[nv];
    int maxflow;

    struct edge{
        int v,w,next;
        edge(){}
        edge(int _v,int _w,int _next):v(_v),w(_w),next(_next){}
    }E[ne];

    void init(int n){
        this->n=n,size=0;
        cl(head,-1);
    }
    void insert(int u,int v,int w){
        E[size]=edge(v,w,head[u]);
        head[u]=size++;
        E[size]=edge(u,0,head[v]);
        head[v]=size++;
    }
    int maxFlow(int src,int des){
        maxflow=0;
        for(int i=0;i<=n;i++){
            dis[i]=gap[i]=0;
            cur[i]=head[i];
        }
        int u=pre[src]=src;
        int aug=0;///or aug=-1
        while(dis[src]<n){
           loop:for(int &i=cur[u];i!=-1;i=E[i].next){
                int v=E[i].v;
                if(E[i].w&&dis[u]==dis[v]+1){
                    aug=min(aug,E[i].w);
                    pre[v]=u;
                    u=v;
                    if(v==des){
                        maxflow+=aug;
                        for(u=pre[u];v!=src;v=u,u=pre[u]){
                            E[cur[u]].w-=aug;
                            E[cur[u]^1].w+=aug;
                        }
                        aug=inf;
                    }
                    goto loop;
                }
            }
            int mdis=n;
            for(int i=head[u];i!=-1;i=E[i].next){
                int v=E[i].v;
                if(E[i].w&&mdis>dis[v]){
                    cur[u]=i;
                    mdis=dis[v];
                }
            }
            if(--gap[dis[u]]==0)break;
            gap[dis[u]=mdis+1]++;
            u=pre[u];
        }
        return maxflow;
    }
};
isap<800,800*800> G;
int mp[maxn][maxn];//存关系图
int f[maxn][maxn];//表示x,y是有关系的
void Floyd(int n){//Floyd 传递关系
    for(int k=1;k<=n;k++){
        for(int i=1;i<=n;i++)if(f[i][k]){
            for(int j=1;j<=n;j++)if(f[k][j]){
                f[i][j]=1;
            }
        }
    }
    for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++)if(f[i][j]){
            for(int k=1;k<=n;k++)if(mp[j][k]){
                mp[i][k]=1;
            }
        }
    }
}
void buildMap(int mid,int k,int n,int m){
    G.init(n*3+1);
    for(int i=1;i<=n;i++){
        G.insert(i,i+n,k);//拆点
        for(int j=1;j<=n;j++){
            if(mp[i][j]){
                G.insert(i,j+n*2,1);
            }
            else {
                G.insert(i+n,j+n*2,1);
            }
        }
    }
    for(int i=1;i<=n;i++){
        G.insert(0,i,mid);
        G.insert(i+n*2,n*3+1,mid);
    }
}
int main(){
    int T;
    scanf("%d",&T);
    while(T--){
        int n,m,k,t;
        scanf("%d%d%d%d",&n,&m,&k,&t);
        cl(mp,0);
        for(int i=0;i<m;i++){
            int x,y;
            scanf("%d%d",&x,&y);
            mp[x][y]=1;
        }
        cl(f,0);
        for(int i=0;i<t;i++){
            int x,y;
            scanf("%d%d",&x,&y);
            f[x][y]=f[y][x]=1;
        }
        Floyd(n);
        int ans=-1;
        int l=0,r=n;
        while(l<=r){
            int mid=l+r>>1;
            buildMap(mid,k,n,m);
            if(G.maxFlow(0,3*n+1)>=n*mid){
                l=mid+1;
                ans=mid;
            }
            else {
                r=mid-1;
            }
        }
        printf("%d\n",ans);
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值