最大权完美匹配:KM算法的优化

KM算法

设二分图的两部分点集分别为 $X=\{X_1, X_2, \ldots, X_n\}$ 和 $Y=\{Y_1, Y_2, \ldots, Y_m\}$, $\left<X_i, Y_j\right>$ 的边权为 $w_{ij}$.

给两部分点集分别赋点权 $\{A_i\}, \{B_i\}$, 使得 $A_i+B_j \ge w_{ij}$. 取等的边的生成子图叫做相等子图。那么相等子图的完美匹配就是最大权匹配。我们需要适当选取权值,使相等子图有完美匹配。

算法流程如下:

  1. 令 $X=\emptyset$, $B_j=0$.
  2. 对于 $k=1, 2, \ldots, n$:
    1. 在 $X$ 中加入 $X_k$, 取 $A_k=\max\{w_{kj}\}$.
    2. 搜索一条从 $X_k$ 到 $Y$ 中未匹配点的交错路。
    3. 如果交错路存在,修改匹配,令 $k \gets k+1$ 重复 $(2)$.
    4. 如果交错路不存在,记搜索树(此时叫做交错树 $M$)顶点集与 $X$ 的交为 $X‘$, 与 $Y$ 的交为 $Y‘$.
      1. 取 $d=\min\{A_i+B_j-w_{ij} \mid X_i \in X‘, \left<X_i, Y_j\right> \notin M\}$.
      2. 将 $X‘$ 中的所有点权减小 $d$, $Y‘$ 中的所有点权增大 $d$. 此时 $A_i+B_j \ge w_{ij}$ 仍然满足,交错树上的边仍然属于相等子图,且至少有一条与交错树相邻的相等子图中的边。
      3. 重复 $(2.2)$.

KM算法需要对每回修改后的子图重新搜索交错路,时间复杂度可达 $O(n^4)$.

优化

由于原交错树仍然是可用的,我们考虑不重新搜索交错路,而是在原交错树上直接扩展新边。

具体地,我们在每次加点的过程中,由起点开始扩展交错树,对于每个 $Y‘$ 中的点,记录它的父结点。对于 $Y_j \in Y \setminus Y‘$, 我们维护 $slack_j=\min\{A_i+B_j-w_{ij}\ \mid X_i \in X‘\}$, 取得最小值的 $X_i$ 是它的准父结点(有多个任取一个)。每次取出 $d=\min\{slack_j \mid Y_j \in Y \setminus Y‘\}$, 用 $d$ 更新点权(顺便更新 $slack$),然后就可以将取得最小值的 $j$(有多个任取一个)加入交错树,$Y_j$ 的父结点就是原准父结点。特别地,当 $d=0$ 是就是继续沿原相等子图扩展。若所加入的 $Y_j$ 还未匹配,则说明已经找到交错路,顺着父结点构成的路径一路修改匹配即可。

在实现上,我们记 $match_j$ 表示 $Y_j$ 的匹配点,$pre_j$ 表示 $Y_j$ 的父结点的匹配点,不存在记为 $0$.

示例代码

示例:假设 $n=m \le 500$, 所有边权和答案绝对值小于 $10^{18}$. 输入 $n$ 和边权,输出 $Y_j$ 的匹配点。

 1 #include <bits/stdc++.h>
 2 const int N=501;
 3 int n, match[N];
 4 bool vis[N];
 5 long long f[N][N], a[N], b[N], slack[N], pre[N];
 6 template<class T1, class T2> bool cmin(T1 &a, const T2 &b)
 7 {
 8     return b<a?(a=b, true):false;
 9 }
10 template<class T1, class T2> bool cmax(T1 &a, const T2 &b)
11 {
12     return a<b?(a=b, true):false;
13 }
14 int main()
15 {
16     scanf("%d", &n);
17     for(int i=1; i<=n; ++i) {
18         for(int j=1; j<=n; ++j)
19             scanf("%d", f[i]+j);
20         a[i]=*std::max_element(f[i]+1, f[i]+n+1);
21     }
22     for(int i=1; i<=n; ++i) {
23         int x=0, cho;
24         memset(vis+1, 0, n);
25         memset(pre+1, 0, n*sizeof(int));
26         memset(slack+1, 63, n*sizeof(long long));
27         match[0]=i;
28         do {
29             int u=match[x];
30             long long min=1e18;
31             vis[x]=true;
32             for(int v=1; v<=n; ++v) {
33                 if(!vis[v]) {
34                     long long t=a[u]+b[v]-f[u][v];
35                     if(cmin(slack[v], t))
36                         pre[v]=x;
37                     if(cmin(min, slack[v]))
38                         cho=v;
39                 }
40             }
41             for(int j=0; j<=n; ++j) {
42                 if(vis[j]) {
43                     a[match[j]]-=min;
44                     b[j]+=min;
45                 } else
46                     slack[j]-=min;
47             }
48             x=cho;
49         } while(match[x]);
50         while(x) {
51             match[x]=match[pre[x]];
52             x=pre[x];
53         }
54     }
55     for(int i=1; i<=n; ++i)
56         printf("%d%c", match[i], " \n"[i==n]);
57     return 0;
58 }

原文地址:https://www.cnblogs.com/nealchen/p/improved-KM.html

时间: 2024-11-09 01:40:52

最大权完美匹配:KM算法的优化的相关文章

二分图最大权值匹配 KM算法 模板

大佬讲的太好了!!!太好了!!! http://www.cnblogs.com/wenruo/p/5264235.html KM算法用来求二分图最大权完美匹配. 本文配合该博文服用更佳:趣写算法系列之--匈牙利算法 本文没有给出KM算法的原理,只是模拟了一遍算法的过程.另,博主水平较差,发现问题欢迎指出,谢谢!!!! 现在有N男N女,有些男生和女生之间互相有好感,我们将其好感程度定义为好感度,我们希望把他们两两配对,并且最后希望好感度和最大. 怎么选择最优的配对方法呢? 首先,每个女生会有一个期

HDU2255-奔小康赚大钱-二分图最大权值匹配-KM算法

二分图最大权值匹配问题.用KM算法. 最小权值的时候把权值设置成相反数 1 /*--------------------------------------------------------------------------------------*/ 2 3 #include <algorithm> 4 #include <iostream> 5 #include <cstring> 6 #include <ctype.h> 7 #include &l

HDU 3722 Card Game(二分图最佳完美匹配+KM算法)

题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3722 1 /* 2 问题 3 将任意的两个字符串进行匹配,使得匹配后权值和最大 4 5 解题思路 6 将任意的字符串的权值计算出来,使用KM算法即可. 7 */ 8 #include<cstdio> 9 #include<cstring> 10 #include<algorithm> 11 using namespace std; 12 13 const int maxn

hdu2255 奔小康赚大钱 km算法解决最优匹配(最大权完美匹配)

/** 题目:hdu2255 奔小康赚大钱 km算法 链接:http://acm.hdu.edu.cn/showproblem.php?pid=2255 题意:lv 思路:最优匹配(最大权完美匹配) km算法 模板来自:http://www.cnblogs.com/wenruo/p/5264235.html 如果是求最小权完美匹配,那么将所有权值取相反数,然后求得最大权,输出最大权的相反数即可. */ #include <iostream> #include <cstring> #

【二分图最大权完美匹配】【KM算法模板】【转】

[文章详解出处]https://www.cnblogs.com/wenruo/p/5264235.html KM算法是用来求二分图最大权完美匹配的.[也就算之前的匈牙利算法求二分最大匹配的变种??] 这里就贴一下模板代码..2333... 1 #include <iostream> 2 #include <cstring> 3 #include <cstdio> 4 #include <algorithm> 5 using namespace std; 6

HDU 3488 Tour (最大权完美匹配)【KM算法】

<题目链接> 题目大意:给出n个点m条单向边边以及经过每条边的费用,让你求出走过一个哈密顿环(除起点外,每个点只能走一次)的最小费用.题目保证至少存在一个环满足条件. 解题分析: 因为要求包含所有点一次的环,我们不难发现,这个环中的每个点的出度和入度均为1,所以我们不妨将每个点进行拆点,将所有点的出度和入度分为两部分.因为该环需要包括所有的点,并且题目需要求该环的最小权值,相当于该带权二分图的每个点都需要被覆盖到,由于本题就转化为求该二分图的最优完美匹配问题.二分图的最优匹配问题求解,我们会想

HDU_2255 二分图最佳完美匹配 KM匈牙利算法

一开始还没看懂这个算法,后来看了陶叔去年的PPT的实例演示才弄懂 用一个lx[]和ly[]来记录X和Y集合中点的权值,有个定理是 lx[i]+ly[j]==w[i][j](边权值) 则该点是最佳匹配,因为首先 那个不等式肯定要>=的,否则就不满足题意了,如果是>则可以去匹配更有价值的边或者把权值降下来让匹配边的潜力更大. 所以只有把握了这个条件,其他就是走一遍最大匹配数.以及up()函数用来在无法匹配的时候,进行其他点的权值降低(也可以说是增广路的搜索)来得到匹配. #include <

带权二分图的最大权匹配 KM算法模版

带权二分图的最大权匹配 KM算法模版 下面是kuangbin大神的模版,已通过西电oj1048的测试 #include<iostream> #include<cstdio> #include<cstring> #include<cstdlib> #include<algorithm> #include<vector> #include<stack> #include<queue> #include<set

POJ 2195 二分图最小权匹配KM算法

本来是打算昨天晚上写的, 昨天网速渣的连CSDN都进不去,没办法 只能现在来写了 先写写对KM算法的理解,KM算法是对每个点设置一个顶标,只有当边长等于两边点的顶标之和的时候才进行增广,这样就能保证得到的一定是最大权匹配. 如果找不到匹配的时候就对交替路中X集合的顶标减少一个d Y集合的顶标增加一个d. 这样两个点都在交替路中的时候x[i]+y[i]的和不边 X在 Y不在的时候x[i]+y[i]减少,可能就会为图增加一对匹配. X不在Y在的时候x[i]+y[i]增加, 原来不在现在依然不在其中.