QAQ现在很不想写题解博客那就来写个算法吧QAQ...

带花树

题目

来看个题...

UOJ79.

某机房里有\(n\)个OIer,其中有\(n\)个男生,\(0\)个女生。现在他们要两两配对。

有\(m\)个关系,每个关系是一个无序对\((a_i,b_i)\),表示这两个人之间愿意配对。

求:最多能配成多少对,并找出一组方案。

说人话:一般图最大匹配。

算法

既然是匹配,我们能不能直接模仿匈牙利算法呢?答案是不可以(废话,可以的话还要什么带花树啊)。

原因是:我们在二分图中,如果dfs找增广路时经过某个点找不到,那么我们可以证明这一轮中这个点的确是无用的(也即,这一轮里所有的增广路都不经过这个点),于是我们就能保证每个点至多走一遍,时间复杂度得到保证。

但是如果是一般图,这个性质不一般成立。比如下图:

图中红线是已匹配的边。那么,如果我从\(1\)开始dfs时先经过\(2\),那么接下来就只能到\(4\)(因为只能走匹配边),然后是\(5,3\),同时我们会给这四个节点都打一个“找不到增广路”的标记。

但是实际上存在\(1\rightarrow3\rightarrow5\rightarrow4\rightarrow2\rightarrow6\)这条增广路。仔细观察我们就能发现:这种现象之所以存在,是由于奇环\(1-3-5-4-2-1\)的存在(如果没有奇环,就变成了二分图匹配,这时候匈牙利算法就是对的)。

那么,对于奇环,有什么后果呢?显然,如果我们dfs出了一个奇环,那么无论环上那个点dfs出了增广路都是可行的。(例如,上图中,如果从\(5\)处dfs出一条增广路,使得\(5\)匹配到别的点且\(3\)成为未盖点,那么可以走\(1-3\)。同样,如果\(5\)被孤立,我们可以走\(1-2-4-5\)来把\(5\)匹配上)

于是,我们可以把一个奇环当做一个点(这个奇环被称为“花”,这就是带花树名字的来历),然后继续找增广路。

形式化的,如果我们在图\(G=(V,E)\)中找到了一个奇环\(v_1-v_2-\dots-v_k-v_1\)(称为“花”),其中\(v_1\)是环上的深度最小的结点,不难证明,\(v_1\)的配偶不在此话中(因为在找到这个花之前所有边组成一个二分图,那么由于\(v_1\)向下dfs/bfs出了一个花,它一定是X结点,只有X结点会向外扩展),且\((v_2,v_3),(v_4,v_5)\dots(v_{k-1},v_{k})\)都是匹配边。那么我们构建一个图

\[\begin{aligned}G'&=(V', E'),\\ V'&=V/\{v_2,v_3\dots,v_k\},\\ E'&=\{(f(a),f(b))|(a,b)\in E, a,b\neq v_i, i=1\dots k\}\end{aligned}
\]

其中\(f(v_i)=v_1, f(a)=a(a\neq v_j), i, j=1,2,\dots,k\)

并且原本\(G\)中的所有匹配除掉\((v_2,v_3),(v_4,v_5)\dots(v_{k-1},v_{k})\)构成\(G'\)的一个匹配。

那么,\(G\)中存在增广路\(\Leftrightarrow\)\(G'\)中存在增广路。

证明:

\(\Rightarrow\):对于\(G\)中的任意一条增广路,若其不经过这朵花,那么在\(G'\)中也存在这条增广路;否则,令这条从\(s\)开始的增广路上的最后一个在花上的点为\(v_j\),那么这条增广路形如 \(s\leadsto v_j \leadsto t\),我们在\(G'\)上构造如下增广路:先从\(s \leadsto v_1 \leadsto t\),其中第一段路程沿着bfs/dfs树走,第二段路程沿着原图中的增广路走,唯一不同的是\(v_j\)变成了\(v_1\)(这是合法的,因为所有从\(v_j\)出发的边都被连到了\(v_1\)上,而且我们根据所有\(v\)都是已盖点可以知道\(v_j\)出发的边是非匹配边)。

\(\Leftarrow\):对于\(G'\)中的一条增广路,若它不经过\(v_1\),则\(G\)中也存在;否则,设这条增广路为\(s\leadsto v_1\rightarrow x \leadsto t\)(\(x\)可能等于\(t\)),根据\(E'\)的定义存在\((v_i, x)\in E\),从而我们构造\(G\)中的增广路:\(s\leadsto v_1\leadsto v_i \rightarrow x \leadsto t\),其中第一段和第三段不变(因为增广路上\(v_1\)至多出现1次,所以这两段在\(G\)中存在),第二段是在花里走(或者精确一点,若\(i\)是奇数,走\(v_1\rightarrow v_2 \dots v_i\),否则走\(v_1\rightarrow v_k \dots v_i\)。证毕。

bfs时,我们可以\(O(n)\)求出LCA并\(O(kn)\)缩花,从而单次bfs至多\(O(n^2)\),总复杂度至多\(O(n^3)\)。

实现上,我们不实际缩点,而是对于每个点维护一个\(fa\),表示它所处的最大的花的LCA(就是\(v_1\))。由于花里可能还有花,这个\(fa\)要用并查集维护。在证明中构造增广路是通过判断\(i\)奇偶性,但实际上我们可以直接维护每个点要往哪边走,也即维护一个\(link_i\)表示如果\(i\)失配要和谁匹配(例如,\(link_{v_2}=v_1,link_{v_3}=v_4\))。找LCA的时候直接暴力\(O(n)\),但要注意只找每个并查集的根节点(因为非根节点都缩到花里了);缩花时要注意如果两个点已经在一朵花里就不要再缩了。

代码

#include <algorithm>
#include <cstdio>
const int N = 550;
const int M = 250050;
int pre[N], nxt[M], to[M], cnt, n;
int vis[N], fa[N], link[N], mate[N];
int que[N], head, tail;
int ss[N], time;
inline void addEdge(int x, int y) {
nxt[cnt] = pre[x];
to[pre[x] = cnt++] = y;
}
int Find(int x) { return fa[x] == x ? x : fa[x] = Find(fa[x]); }
int LCA(int x, int y) {
++time;
while (ss[x] != time) {
if (x) {
ss[x] = time;
x = Find(link[mate[x]]);
}
std::swap(x, y);
}
return x;
}
void flower(int x, int y, int p) {
while (Find(x) != p) {
link[x] = y;
fa[y = mate[x]] = fa[x] = p;
if (vis[y] == 1)
vis[que[tail++] = y] = 2;
x = link[y];
}
}
bool match(int x) {
head = tail = 0;
for (int i = 1; i <= n; ++i)
vis[fa[i] = i] = 0;
vis[que[tail++] = x] = 2;
while (head != tail) {
x = que[head++];
for (int i = pre[x]; ~i; i = nxt[i]) {
int u = to[i];
if (!vis[u]) {
vis[u] = 1;
link[u] = x;
if (mate[u])
vis[que[tail++] = mate[u]] = 2;
else {
while (u) {
x = mate[link[u]];
mate[mate[u] = link[u]] = u;
u = x;
}
return true;
}
} else if (vis[u] == 2 && Find(u) != Find(x)) {
int p = LCA(x, u);
flower(x, u, p);
flower(u, x, p);
}
}
}
return false;
}
int main() {
int m, x, y, ans = 0;
scanf("%d%d", &n ,&m);
std::fill(pre + 1, pre + n + 1, -1);
while (m--) {
scanf("%d%d", &x, &y);
addEdge(x, y);
addEdge(y, x);
if (!mate[x] && !mate[y])
mate[mate[x] = y] = x, ++ans;
}
for (int i = 1; i <= n; ++i)
if (!mate[i] && match(i))
++ans;
printf("%d\n", ans);
for (int i = 1; i <= n; ++i)
printf("%d ", mate[i]);
return 0;
}

WC前的颓废——带花树的更多相关文章

  1. WC前的小计划

    写在前面的.. 要去WC了好开心的呢.. 但是之前荒废了好多时间呢.. 好吧从明天开始加紧训练,目标是:WC前bzoj300t..(现在是260呢..) 开始吧 来看看完成情况: 40/40 [201 ...

  2. 【learning】一般图最大匹配——带花树

    问题描述 ​ 对于一个图\(G(V,E)\),当点对集\(S\)满足任意\((u,v)\in S\),均有\(u,v\in V,(u,v)\in E\),且\(S\)中没有点重复出现,我们称\(S\) ...

  3. [转]带花树,Edmonds's matching algorithm,一般图最大匹配

    看了两篇博客,觉得写得不错,便收藏之.. 首先是第一篇,转自某Final牛 带花树……其实这个算法很容易理解,但是实现起来非常奇葩(至少对我而言). 除了wiki和amber的程序我找到的资料看着都不 ...

  4. HDOJ 4687 Boke and Tsukkomi 一般图最大匹配带花树+暴力

    一般图最大匹配带花树+暴力: 先算最大匹配 C1 在枚举每一条边,去掉和这条边两个端点有关的边.....再跑Edmonds得到匹配C2 假设C2+2==C1则这条边再某个最大匹配中 Boke and ...

  5. 【Learning】带花树——一般图最大匹配

    一般图最大匹配--带花树 问题 ​ 给定一个图,求该图的最大匹配.即找到最多的边,使得每个点至多属于一条边. ​ 这个问题的退化版本就是二分图最大匹配. ​ 由于二分图中不存在奇环,偶环对最大匹配并无 ...

  6. [BZOJ]4405: [wc2016]挑战NPC(带花树)

    带花树模板 #include<cstdio> #include<cstring> #include<algorithm> using namespace std; ...

  7. 【XSY2774】学习 带花树

    题目描述 给你一个图,求最大匹配. 边的描述方式很特殊,就是一次告诉你\(c_i\)个点:\(d_1,d_2,\ldots,d_{c_i}\),表示这些点两两之间都有连边,也就是说,这是一个团.总共有 ...

  8. HDU 4687 Boke and Tsukkomi (一般图最大匹配)【带花树】

    <题目链接> 题目大意: 给你n个点和m条边,每条边代表两点具有匹配关系,问你有多少对匹配是冗余的. 解题分析: 所谓不冗余,自然就是这对匹配关系处于最大匹配中,即该匹配关系有意义.那怎样 ...

  9. URAL 1099 Work Scheduling (一般图最大匹配) 模板题【带花树】

    <题目链接> <转载于 >>>  > 题目大意: 给出n个士兵,再给出多组士兵之间两两可以匹配的关系.已知某个士兵最多只能与一个士兵匹配.求最多能够有多少对匹 ...

随机推荐

  1. Python数据结构之序列及其操作

    数据结构是计算机存储,组织数据的方式.数据结构是指相互之间存在一种或多种特定关系的数据元素的集合. 在Python中,最基本的数据结构为序列(sequence).序列中的每个元素都有编号:从0开始递增 ...

  2. win7 wifi sharing

    1.启用并设定虚拟WiFi网卡: netsh wlan set hostednetwork mode=allow ssid=mywifi key=12345678 此命令有三个参数,mode:是否启用 ...

  3. js数字格式化为千分位

    方法1: 浏览器自带的一个方法 const num=12345.6789 num.toLocaleString();=>"12,345.679" 方法2: 正则匹配 func ...

  4. 思科 ISR路由器登录内置交换模块的方式

    ISR2900/3900系列 登录:Router#service-module gigabitethernet1/0 session 退出: control+shift+6 x disconnect ...

  5. es6里class类

    /** * Created by issuser on 2018/11/27. *///如果静态方法包含this关键字,这个this指的是类,而不是实例./** (1)类的实例属性 1.类的实例属性可 ...

  6. Mac下使用Eclipse的Show in Terminal提示command not found: mvn

    在Mac下一般配置了Maven的环境变了一般都不会提示,但是如果使用zsh的扩展之后,系统默认的环境变量配置文件会发生变化,尤其使用Eclipse打开终端时,默认不会去读取用户目录下的~/.bashr ...

  7. Mac 10.12为打开终端增加快捷键(转)

    1.在实用工具中打开Automator.app 2.选择新建,然后选择服务 3.服务收到选择为没有输入 然后在左边侧栏中双击Run AppleScript(有些系统会显示运行 AppleScript) ...

  8. IE haslayout的理解与bug修复

    要想更好的理解 css, 尤其是 IE 下对 css 的渲染,haslayout 是一个非常有必要彻底弄清楚的概念.大多 IE下的显示错误,就是源于 haslayout 什么是 haslayout ? ...

  9. ThreadPoolExecutor 杂记

    When a new task is submitted in method execute(Runnable), and fewer than corePoolSize threads are ru ...

  10. 用尾递归和普通递归实现n!算法,二者比较

    尾递归 - Tail Recursion尾递归是针对传统的递归算法而言的, 传统的递归算法在很多时候被视为洪水猛兽. 它的名声狼籍, 好像永远和低效联系在一起.尾递归就是从最后开始计算, 每递归一次就 ...