zoukankan      html  css  js  c++  java
  • POJ2112 Optimal Milking(二分图多重匹配)

    题意:

    K个产奶机,C头奶牛,每个产奶机最多可供M头奶牛使用;并告诉了产奶机、奶牛之间的两两距离Dij(0<=i,j<K+C)。

    问题:如何安排使得在任何一头奶牛都有自己产奶机的条件下,奶牛到产奶机的最远距离最短?最短是多少?

    思路:

    二分图多重匹配

    先用floyd求最短距离

    然后跟上一个题一样

    不过是奶牛与可达的产奶机建边

    依旧是二分答案

    /* ***********************************************
    Author        :devil
    Created Time  :2016/5/17 22:7:52
    ************************************************ */
    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    #include <vector>
    #include <queue>
    #include <set>
    #include <map>
    #include <string>
    #include <cmath>
    #include <stdlib.h>
    using namespace std;
    const int N=235;
    const int M=35;
    const int inf=0x3f3f3f3f;
    int n,m,c,mid,mp[N][N];
    bool vis[M];
    vector<int>eg[N];
    bool Find(int u)
    {
        for(int i=0; i<eg[u].size(); i++)
        {
            int v=eg[u][i];
            if(!vis[v])
            {
                vis[v]=1;
                if(eg[v].size()<m)
                {
                    eg[v].push_back(u);
                    return 1;
                }
                for(int j=0; j<eg[v].size(); j++)
                {
                    if(Find(eg[v][j]))
                    {
                        eg[v][j]=u;
                        return 1;
                    }
                }
            }
        }
        return 0;
    }
    bool maxmatch()
    {
        for(int i=1; i<=n+c; i++)
            eg[i].clear();
        for(int i=n+1; i<=n+c; i++)
            for(int j=1; j<=n; j++)
                if(mp[i][j]<=mid)
                    eg[i].push_back(j);
        for(int i=n+1; i<=n+c; i++)
        {
            memset(vis,0,sizeof(vis));
            if(!Find(i)) return 0;
        }
        return 1;
    }
    void floyd()
    {
        for(int k=1; k<=n+c; k++)
            for(int i=1; i<=n+c; i++)
                for(int j=1; j<=n+c; j++)
                    if(mp[i][j]>mp[i][k]+mp[k][j])
                        mp[i][j]=mp[i][k]+mp[k][j];
    }
    int main()
    {
        //freopen("in.txt","r",stdin);
        while(~scanf("%d%d%d",&n,&c,&m))
        {
            for(int i=1; i<=n+c; i++)
                for(int j=1; j<=n+c; j++)
                {
                    scanf("%d",&mp[i][j]);
                    if(!mp[i][j]) mp[i][j]=inf;
                }
            floyd();
            int l=0,r=(n+c)*200;
            while(l<r)
            {
                mid=(l+r)>>1;
                if(maxmatch()) r=mid;
                else l=mid+1;
            }
            printf("%d
    ",r);
        }
        return 0;
    }
  • 相关阅读:
    echarts 地图 动态 展示 结合css+js
    优化之误!
    SQL Server 运行计划操作符具体解释(3)——计算标量(Compute Scalar)
    NHibernate概括
    C++的IO操作
    (数据结构整理)NJUPT1054
    nginx模块开发
    Html学习(三) 分类学习
    android 屏幕适配
    【POJ 1845】 Sumdiv (整数唯分+约数和公式+二分等比数列前n项和+同余)
  • 原文地址:https://www.cnblogs.com/d-e-v-i-l/p/5503366.html
Copyright © 2011-2022 走看看