zoukankan      html  css  js  c++  java
  • Golden Tiger Claw(二分图)

    Golden Tiger Claw

    题意

    找到和最小的两个序列a,b满足对于任意i,j有a[i]+b[j]>=c[i][j](矩阵c给出)。

    solution

    裸的二分图就水过了……

    #include<cstdio>
    #include<cstring>
    #include<cmath>
    #include<algorithm>
    #define clr(a,m) memset(a,m,sizeof(a))
    #define rep(i,a,b) for(int i=a;i<=b;i++)
    using namespace std;
    
    const int MAXN=555;
    const int INF =1e9;
    const double eps=1e-10;
    
    int gap[MAXN][MAXN];
    int Lx[MAXN],Ly[MAXN],slack[MAXN];
    int left[MAXN],n;
    bool S[MAXN],T[MAXN];
    
    void read()
    {
        rep(i,1,n)
            rep(j,1,n)
                scanf("%d",&gap[i][j]);
    }
    
    bool match(int u)
    {
        S[u]=true;
        rep(v,1,n)
            if(!T[v]){
                int tmp=Lx[u]+Ly[v]-gap[u][v];
                if(tmp==0){
                    T[v]=true;
                    if(!left[v]||match(left[v])){
                        left[v]=u;
                        return true;
                    }
                }else slack[v]=min(slack[v],tmp);
            }
        return false;
    }
    
    void update()
    {
        int a=INF;
        rep(v,1,n)
            if(!T[v])
                a=min(a,slack[v]);
        rep(i,1,n){
            if(S[i])Lx[i]-=a;
            if(T[i])Ly[i]+=a;
        }
    }
    
    void KM()
    {
        rep(i,1,n){
            left[i]=Ly[i]=0;
            Lx[i]=-INF;
            rep(j,1,n)
                Lx[i]=max(Lx[i],gap[i][j]);
        }
        rep(i,1,n){
            rep(j,1,n)
                slack[j]=INF;
            while(1)
            {
                rep(j,1,n)
                    S[j]=T[j]=0;
                if(match(i))
                    break;
                else
                    update();
            }
        }
    }
    
    void print()
    {
        int ans=0;
        rep(i,1,n){
            ans+=Lx[i];
            if(i!=1)printf(" ");
            printf("%d",Lx[i]);
    
        }
        printf("
    ");
        rep(i,1,n){
            ans+=Ly[i];
            if(i!=1)printf(" ");
            printf("%d",Ly[i]);
        }
        printf("
    ");
        printf("%d
    ",ans);
    }
    
    int main()
    {
        while(~scanf("%d",&n))
        {
            read();
            KM();
            print();
        }
        return 0;
    }
    
  • 相关阅读:
    Js 循环 forEach, map, for, for...in, for...of
    es6 let const
    es6 Symbol类型
    es6 Reflect 与 Proxy
    es6 Map&Set
    es6箭头函数
    es6数组Arrary
    学写网站(一)前端配置之安装nvm、node、npm
    python获取当前执行代码所在文件的地址、主程序所在地址
    scrapy中的选择器下载中间件downloadmiddlewares
  • 原文地址:https://www.cnblogs.com/rui-4825/p/12870759.html
Copyright © 2011-2022 走看看