HDU 2853 最大匹配&KM模板

http://acm.hdu.edu.cn/showproblem.php?pid=2853

这道题初看了没有思路,一直想的用网络流如何解决

参考了潘大神牌题解才懂的

最大匹配问题KM

还需要一些技巧来解决最小变动,

做法是:把原先的邻接矩阵每个数扩大k倍(k>n)

为了突出原先的选择,也就是同等情况下优先选择原来的方案

给原来的方案对应矩阵内的数据+1

那么

最终得出的最大匹配值/k=真实的最大匹配

最终得出的最大匹配值%k=原来的方案采用了几个

这里的KM留下来做模板

/*
二分图最佳匹配 (kuhn munkras 算法 O(m*m*n)).
邻接矩阵形式 。  返回最佳匹配值,传入二分图大小m,n
邻接矩阵 mat ,表示权,match1,match2返回一个最佳匹配,为匹配顶点的match值为-1,
一定注意m<=n,否则循环无法终止,最小权匹配可将全职取相反数。
初始化:  for(i=0;i<MAXN;i++)
             for(j=0;j<MAXN;j++) mat[i][j]=-inf;
对于存在的边:mat[i][j]=val;//注意不能负值
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#define maxn 55
#define inf 1000000000
#define _clr(x) memset(x,-1,sizeof(int)*maxn)
using namespace std;
int donser[maxn][maxn];
int match1[maxn],match2[maxn];
int km(int m,int n,int mat[][maxn],int *match1,int *match2)
{
        int s[maxn],t[maxn],ak[maxn],ac[maxn];
    int p,q,i,j,k,ret=0;
    for(i=0;i<m;i++)
    {
        ak[i]=-inf;
        for(j=0;j<n;j++)
            ak[i]=mat[i][j]>ak[i]?mat[i][j]:ak[i];
        if(ak[i]==-inf)  return -1;
    }
    for(i=0;i<n;i++)
        ac[i]=0;
    _clr(match1);
    _clr(match2);
    for(i=0;i<m;i++)
    {
        _clr(t);
        p=0;q=0;
        for(s[0]=i;p<=q&&match1[i]<0;p++)
        {
            for(k=s[p],j=0;j<n&&match1[i]<0;j++)
            {
                if(ak[k]+ac[j]==mat[k][j]&&t[j]<0)
                {
                    s[++q]=match2[j];
                    t[j]=k;
                    if(s[q]<0)
                    {
                        for(p=j;p>=0;j=p)
                        {
                            match2[j]=k=t[j];
                            p=match1[k];
                            match1[k]=j;
                        }
                    }
                }
            }
        }
        if(match1[i]<0)
        {
            i--;
            p=inf;
            for(k=0;k<=q;k++)
            {
                for(j=0;j<n;j++)
                {
                    if(t[j]<0&&ak[s[k]]+ac[j]-mat[s[k]][j]<p)
                       p=ak[s[k]]+ac[j]-mat[s[k]][j];
                }
            }
            for(j=0;j<n;j++)
               ac[j]+=t[j]<0?0:p;
            for(k=0;k<=q;k++)
               ak[s[k]]-=p;
        }
    }
    for(i=0;i<m;i++)
        ret+=mat[i][match1[i]];
    return ret;
}
int main()
{
    int n,m,i,j;
    while(~scanf("%d%d",&n,&m))
    {
        int k=n+1,t,num=0;
        for(i=0;i<n;i++)
        {
            for(j=0;j<m;j++)
            {
                scanf("%d",&t);
                donser[i][j]=t*k;
            }
        }
        for(i=0;i<n;i++)
        {
            scanf("%d",&t);
            //cout<<i<<" "<<t-1<<" "<<donser[i][t-1]<<endl;
            num+=donser[i][t-1]/k;
            donser[i][t-1]+=1;
        }
        int kk=km(n,m,donser,match1,match2);
        cout<<n-kk%k<<" "<<kk/k-num<<endl;
        memset(donser,0,sizeof(donser));
    }
    return 0;
}
时间: 2024-10-10 10:13:37

HDU 2853 最大匹配&KM模板的相关文章

hdu 2853 Assignment KM算法

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2853 Last year a terrible earthquake attacked Sichuan province. About 300,000 PLA soldiers attended the rescue, also ALPCs. Our mission is to solve difficulty problems to optimization the assignment of t

HDU 2853 Assignment(KM最大匹配好题)

HDU 2853 Assignment 题目链接 题意:现在有N个部队和M个任务(M>=N),每个部队完成每个任务有一点的效率,效率越高越好.但是部队已经安排了一定的计划,这时需要我们尽量用最小的变动,使得所有部队效率之和最大.求最小变动的数目和变动后和变动前效率之差. 思路:对于如何保证改变最小,没思路,看了别人题解,恍然大悟,表示想法非常机智 试想,如果能让原来那些匹配边,比其他匹配出来总和相同的权值还大,对结果又不影响,那就简单了,这个看似不能做到,其实是可以做到的 数字最多选出50个,所

hdu 2063 过山车(二分图匹配最大匹配数模板)

过山车 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 10776    Accepted Submission(s): 4748 Problem Description RPG girls今天和大家一起去游乐场玩,终于可以坐上梦寐以求的过山车了.可是,过山车的每一排只有两个座位,而且还有条不成文的规矩,就是每个女生必须找个个男生做par

HDU 2853 &amp;&amp; HDU 3315

http://acm.hdu.edu.cn/showproblem.php?pid=2853 题意:给一个n-m二分图,边权用一个n*m的矩阵表示,给出初始匹配,求二分图完美匹配相比初始匹配改变了几条边以及改变的数值 这类题的主要思想是增加原配边的权值,但又不影响最后结果. 步骤1:观察顶点数,每条边乘一个大于顶点数的数v 步骤2:对于原配边,每边加1(注意步骤2可以保证km()/v的结果与原结果相同) 步骤3:求完美匹配,答案为res,改变的边数=n-res%v(res%v表示完美匹配中有多少

hdu 1083 最大匹配

题意:N个学生 P 个课程  求最大匹配 3 3 //学生 课程 3 1 2 3 //课程1 匹配学生1 2 3 2 1 2 1 1典型的匹配没什么好说的 ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 #include<iostream

hdu 1179 最大匹配

题意:n个ren m个棍子 每个棍子可以与i个人结合 问最大有多少个结合#include<iostream> #include<cmath> using namespace std; int map[111][111]; int fa[111]; int v[111]; int n,m; int dfs(int x) { for(int i=1;i<=n;i++) { if(map[x][i]&&!v[i])//回溯时i不能再匹配 { v[i]=1; if(!

hdu 2853 Assignment 费用流

就是本来就给出了一个匹配,然后让你求一个权值最大的匹配,并且和初始匹配变动最小. #include <stdio.h> #include <iostream> #include <string.h> using namespace std; const int N=400; const int MAXE=20000000; const int inf=1<<30; int head[N],s,t,cnt,ans; int d[N],pre[N]; bool

POJ 3528 hdu 3662 三维凸包模板题

POJ 3528题:http://poj.org/problem?id=3528 HDU 3662:http://acm.hdu.edu.cn/showproblem.php?pid=3662 一个是求三维凸包面数,一个是求三维凸包表面积,都是很裸的. 贴代码: #include<stdio.h> #include<algorithm> #include<string.h> #include<math.h> #include<stdlib.h>

HDU 2853 (KM最大匹配)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2853 题目大意:二分图匹配费用流.①最大匹配②最小原配变动 解题思路: 如果去掉第二个要求,那么就是裸KM. 然而加上第二个要求,那么就需要一种新的建图方式. 建图 对于输入矩阵,每一条边,cost扩大K倍($K=n+1$) 对于原配,每一条边cost在扩大K倍基础上+1 KM 统计cost时,直接把cost整除K,然后累加. 并且Hash一下原配边的变动情况. 扩大K倍的作用 准确来说,K倍是为了