题目链接

https://www.luogu.com.cn/problem/P6577

题目大意

给定一个二分图,其左右点的个数各为 \(n\),带权边数为 \(m\),保证存在完美匹配。

求一种完美匹配的方案,使得最终匹配边的边权之和最大。

题目解析

二分图最大权完美匹配,一般用 \(KM\) 算法完成。

基础版的 \(KM\) 算法,是匈牙利算法的改进,依然采用 \(DFS\) 策略,速度较慢。

而改进版的 \(KM\) 算法采用 \(BFS\) 策略,有效提升速度。

点数为 \(n\),边数为 \(m\) 。

时间复杂度: \(O(n^2 m)\)

参考代码

基础版(会超时)

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int INF = 0x3f3f3f3f;
const int N = 505;
int mp[N][N], lx[N], ly[N], visx[N], visy[N], match[N];
int n, m, minz, k; bool dfs(int x, int K)
{
visx[x] = K;
for (int y = 1; y <= n; ++y) {
if (visy[y] != K && mp[x][y] != INF) {
int t = lx[x] + ly[y] - mp[x][y];
if (!t) {
visy[y] = K;
if (!match[y] || dfs(match[y], K)) {
match[y] = x;
return true;
}
}
else minz = min(minz, t);
}
}
return false;
}
void KM()
{
for (int i = 1; i <= n; ++i) {
while (1) {
minz = INF;
if (dfs(i, ++k)) break;
for (int j = 1; j <= n; ++j) {
if (visx[j] == k) lx[j] -= minz;
if (visy[j] == k) ly[j] += minz;
}
}
}
}
int main()
{
scanf("%d%d", &n, &m);
memset(mp, INF, sizeof mp);
for (int i = 1; i <= n; ++i) lx[i] = -INF;
for (int i = 0; i < m; ++i) {
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
mp[u][v] = w;
lx[u] = max(lx[u], w);
}
KM();
ll ans = 0;
for (int i = 1; i <= n; ++i) ans += mp[match[i]][i];
printf("%lld\n", ans);
for (int i = 1; i <= n; ++i) printf("%d ", match[i]);
putchar('\n');
return 0;
}

改进版

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int INF = 0x3f3f3f3f;
const int N = 505;
int mp[N][N], lx[N], ly[N], visy[N], matchy[N], pr[N], slack[N];
int n, m, minz, k; void bfs(int x, int K)
{
int y = 0, yy = 0;
memset(pr, 0, sizeof pr);
memset(slack, INF, sizeof slack);
matchy[y] = x;
while (true)
{
x = matchy[y];
visy[y] = K;
for (int i = 1; i <= n; ++i)
{
if (visy[i] == K || mp[x][i] == -INF) continue;
int t = lx[x] + ly[i] - mp[x][i];
if (slack[i] > t)
{
slack[i] = t;
pr[i] = y;
}
}
minz = INF;
for (int i = 1; i <= n; ++i) {
if (visy[i] != K && slack[i] < minz) {
minz = slack[i];
yy = i;
if (!minz) break;
}
}
if (minz) {
for(int i = 0; i <= n; ++i)
{
if (visy[i] == K) lx[matchy[i]] -= minz, ly[i] += minz;
else slack[i] -= minz;
}
}
y = yy;
if (!matchy[y]) break;
}
while (y) {
matchy[y] = matchy[pr[y]];
y = pr[y];
}
}
void KM()
{
for (int i = 1; i <= n; ++i) {
bfs(i, ++k);
}
}
int main()
{
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= n; ++j) {
mp[i][j] = -INF;
}
}
for (int i = 1; i <= n; ++i) lx[i] = -INF;
for (int i = 0; i < m; ++i) {
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
mp[u][v] = w;
lx[u] = max(lx[u], w);
}
KM();
ll ans = 0;
for (int i = 1; i <= n; ++i) ans += mp[matchy[i]][i];
printf("%lld\n", ans);
for (int i = 1; i <= n; ++i) printf("%d ", matchy[i]);
putchar('\n');
return 0;
}

感谢支持!

【模板】二分图最大权完美匹配(KM算法)/洛谷P6577的更多相关文章

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

    hdu2255模板题 KM是什么意思,详见百度百科. 总之知道它可以求二分图最大权完美匹配就可以了,时间复杂度为O(n^3). 给张图. 二分图有了边权,求最大匹配下的最大权值. 所以该怎么做呢?对啊 ...

  2. 二分图最大权完美匹配KM算法

    KM算法二分图 KM求得二分图与普通二分图的不同之处在于:此二分图的每条边(男生女生)上都附了权值(好感度).然后,求怎样完美匹配使得权值之和最大. 这,不止一般的麻烦啊. 可以通过一个期望值来求. ...

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

    KM算法详解+模板 大佬讲的太好了!!!太好了!!! 转载自:http://www.cnblogs.com/wenruo/p/5264235.html KM算法用来求二分图最大权完美匹配. 本文配合该 ...

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

    二分图最大权值匹配问题.用KM算法. 最小权值的时候把权值设置成相反数 /*-------------------------------------------------------------- ...

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

    [文章详解出处]https://www.cnblogs.com/wenruo/p/5264235.html KM算法是用来求二分图最大权完美匹配的.[也就算之前的匈牙利算法求二分最大匹配的变种??] ...

  6. Solution -「洛谷 P6577」「模板」二分图最大权完美匹配

    \(\mathcal{Description}\)   Link.   给定二分图 \(G=(V=X\cup Y,E)\),\(|X|=|Y|=n\),边 \((u,v)\in E\) 有权 \(w( ...

  7. 二分图学习记 之 KM算法 二分图最大权完美匹配。

    前置知识 :匈牙利算法 首先有这样一张图,求这张图的最大权完美匹配. 当然如果你不想看这些渣图的话,您可以转到 洛谷 运动员最佳匹配问题 下面我来强行解释一下KM算法 左边一群妹子找汉子,但是每个妹子 ...

  8. uva 1411 Ants (权值和最小的完美匹配---KM算法)

    uva 1411 Ants Description Young naturalist Bill studies ants in school. His ants feed on plant-louse ...

  9. hdu 3722 二分图 最优完备匹配 KM算法

    这题只要想到是最优完备匹配就行了: 题意:给出n个字符串,若两两相连,将前一个反置添加到后一个后面,相连的值为两个字串从头开始相等的字符个数: 问如何匹配得出最大值: 思路:建图,套模板. 代码: # ...

随机推荐

  1. 『学了就忘』Linux基础 — 10、VMware虚拟机中克隆的使用

    目录 1.什么是克隆 2.克隆的两种类型 (1)完整克隆 (2)链接克隆 3.克隆操作 步骤一:克隆虚拟机 步骤二:进行克隆导向 3.快照与克隆的区别 4.镜像的管理 快照和克隆是VMware中两个非 ...

  2. Python ImportError: cannot import name ABC

    Python 3.5.2 测试可以运行 import sys from abc import ABC,abstractmethod class MyBase(ABC): @abstractmethod ...

  3. 第12课 OpenGL 显示列表

    显示列表: 想知道如何加速你的OpenGL程序么?这一课将告诉你如何使用OpenGL的显示列表,它通过预编译OpenGL命令来加速你的程序,并可以为你省去很多重复的代码. 这次我将教你如何使用显示列表 ...

  4. Linux安装部署Zabbix

    Zabbix 是一个企业级的分布式开源监控方案,能够监控各种网络参数以及服务器健康性和完整性的软件.Zabbix使用灵活的通知机制,允许用户为几乎任何事件配置基于邮件的告警.这样可以快速反馈服务器的问 ...

  5. Linux下软链接与硬链接的区别

    由于下面会说到inode,所以如果没有了解过,请务必搞懂inode的真正含义,厚颜无耻的推荐我的一篇博客:Linux磁盘与文件系统管理 如果我们在系统中新建一个文件,我们看到的文件名实际上只是表面现象 ...

  6. Kali-Linux 2020如何设置中文

    话不多说,直接上步骤 首先,想要修改系统默认语言普通用户是办不到的,这个时候就需要切换为root用户在终端输入 sudo su(切换用户指令,后面不加用户名就默认切换为root) 输入管理员密码后就像 ...

  7. C# | VS2019连接MySQL的三种方法以及使用MySQL数据库教程

    本文将介绍3种添加MySQL引用的方法,以及连接MySQL和使用MySQL的教程 前篇:Visual Studio 2019连接MySQL数据库详细教程 \[QAQ \] 第一种方法 下载 Mysql ...

  8. maven中的distributionManagement的作用

    mvn install  会将项目生成的构件安装到本地Maven仓库,mvn deploy 用来将项目生成的构件分发到远程Maven仓库. 本地Maven仓库的构件只能供当前用户使用,在分发到远程Ma ...

  9. AsExpandable EF多条件查询

    我个人学习新技术有一个方法,如果遇到问题会根据以前的经验来寻找一些类似的解决方法.有人会说,如果这个问题在你的学习或者工作生涯中都没有遇到过呢?很简单,通过搜索资料或查阅相关书籍学习别人的经验. 在如 ...

  10. JS表格显示时间格式

    <!-- JS代码区 --> <script type='text/javascript'> $(function() { var grid_selector23 = &quo ...