首页 > 分享 > 带花树算法学习笔记

带花树算法学习笔记

带花树算法学习笔记

难得yyb写了一个这么正式的标题

Q:为啥要学带花树这种东西啊?
A:因为我太菜了,要多学点东西才能不被吊打
Q:为啥要学带花树这种东西啊?
A:因为我做自己的专题做不动了,只能先去“预习”ppl的专题了
Q:为啥要学带花树这种东西啊?
A:因为可以用来做题啊,比如某WC题目

先推荐一个很皮很皮的带花树讲解:
戳这里嗷

QaQ
言归正传
带花树的算法用来解决一般图的最大匹配问题
说起来,是不是想起来网络流里面的最小路径覆盖?
或者二分图的最大匹配的问题?
的确,带花树解决一般图的最大匹配问题类似于这些东西。
但是肯定是有不同的。

比方说:
我们用匈牙利的思路来解决一般图
我们是可以很容易就让算法挂掉的
只需要一个奇环就可以啦
(让我偷张图片过来)

看见没有
有了一个奇环,在匹配的时候黑白就会翻转过来。
所以我们当然不能直接用匈牙利来做。

但是,这样的问题当然需要解决,
所以就有了带花树算法。
你可以理解为:
带花树算法=匈牙利算法+处理奇环

因为不打算长篇大论,
我按照带花树的步骤来写写这个算法。
(随时对比匈牙利算法)

匈牙利算法第一步:找到一个未被匹配的点,从这个点开始匹配
带花树算法第一步:找到一个未被匹配的点,从这个点开始匹配

貌似没有区别。。。
接下来匈牙利算法会用dfs" role="presentation">dfs来寻找增广路
带花树算法使用bfs" role="presentation">bfs
将当前点丢进队列里面
我们将他染个色,比如说黑色
然后开始bfs" role="presentation">bfs
首先取出队首的黑点u" role="presentation">u
找找和它相邻的点v,(u,v)∈E" role="presentation">v,(u,v)∈E
如果v" role="presentation">v是白点并且在当前的这一次匹配中已经被访问过,则不管这个点
否则,如果当前点v" role="presentation">v没有被访问过,并且v" role="presentation">v没有匹配点
那么就是找到了一条增广路
记录每一个点的前驱pre" role="presentation">pre,每个点的匹配点match" role="presentation">match
从当前的点v" role="presentation">v开始,每个点都和他的前驱两两匹配
沿着增广路全部修改回去就行了,
然后这一次的匹配结束。(这个跟匈牙利是一样的啊)
如果这个点已经有匹配点的话,则去尝试能否修改它的匹配点
因此,这个时候把v" role="presentation">v的前驱置为u" role="presentation">u,然后把v" role="presentation">v的匹配点丢进队列里面。(这也是和匈牙利一样的啊)
继续bfs" role="presentation">bfs,尝试能否修改它的匹配点。

对于上面的情况,明显和匈牙利算法是一模一样的,
但是出现了匈牙利不能解决的情况,也就是奇环。

如果当前黑点u" role="presentation">u的相邻点扩展出来了一个黑点v" role="presentation">v,
意味着u−v−u" role="presentation">u−v−u构成了一个奇环
那么我们就要缩环啦,这就是带花树算法的重点。

对于一个奇环,它的点的个数一定是2k+1" role="presentation">2k+1的形式
意味着,在奇环内最多只有k" role="presentation">k组匹配,
同时,一定有一个点会向外匹配(匹配点不在环内)
现在,如果我们把整个奇环都看成一个点
如果某个增广路找到了奇环上去,我们一定能够重置奇环内的匹配
无非是把增广路找到的奇环上的那个点和增广路上的其他点匹配。
然后奇环剩下的2k" role="presentation">2k个点两两匹配。

所以,我们可以直接把奇环看成一个点来缩,这个就是开花啦
如果增广路找到了奇环上,我们就把奇环展开重新更新一下匹配就好。

可是,问题是,怎么缩奇环???
我们额外维护一个并查集,将同朵花中的节点在并查集中合并
我们先求出他们的最近花祖先
这个要怎么理解?
我们的匹配(match" role="presentation">match)和前驱(pre" role="presentation">pre)都是边
如果把已经缩好的奇环都看成一个点
那么,这些边和点,就是一棵树。
假设现在出现了u−v" role="presentation">u−v这条边
意味着在树上出现了一个基环(当然也是奇环)
那么,从当前的u,v" role="presentation">u,v所在的奇环开始(如果只有一个点就是它自己啦)
不断的向上走交替地沿着match" role="presentation">match和pre" role="presentation">pre边向上
当然了,每次走当然要走到他所在的奇环(并查集的根节点)所代表的那个位置啦(这是朴素的、暴力的lca" role="presentation">lca求法)

所以求lca" role="presentation">lca的代码如下:

int lca(int u,int v) {++tim;u=getf(u);v=getf(v);while(dfn[u]!=tim){dfn[u]=tim;u=getf(pre[match[u]]);if(v)swap(u,v);}return u; }

dfn" role="presentation">dfn就是一个标记而已,你在向上跳的时候一边跳一边打标记
如果你在跳完另外一个点后发现这个位置已经被打了标记,
那么就意味着这个点就是lca" role="presentation">lca啦

好的,我们求出来了LCA" role="presentation">LCA,考虑怎么缩环(开花)
先上代码我再来解释

void Blossom(int x,int y,int w) {while(getf(x)!=w){pre[x]=y,y=match[x];if(vis[y]==2)vis[y]=1,Q.push(y);if(getf(x)==x)f[x]=w;if(getf(y)==y)f[y]=w;x=pre[y];} }

x,y" role="presentation">x,y是要开花的奇环的两个点(也就是上面的u,v" role="presentation">u,v)
w" role="presentation">w是他们的LCA" role="presentation">LCA
此时x,y" role="presentation">x,y之间可以匹配,但是他们都是黑点。

因为整朵花缩完都是一个黑点
因此,我们把x−>lca" role="presentation">x−>lca,v−>lca" role="presentation">v−>lca的路径全部处理即可
因为两部分相同,因此只需要写一个Blossom" role="presentation">Blossom函数
看看这个开花是怎么执行的
首先把x,y" role="presentation">x,y用pre" role="presentation">pre连接起来(默认一朵花中未匹配的点就是lca" role="presentation">lca,也就是花根)
然后沿着x" role="presentation">x(或者y" role="presentation">y)向上一个个点往上跳
如果跳到某个点是白点,但是花中的所有点都是黑点
所以把白点暴力染黑,然后丢进队列中增广

在跳的过程中,很可能中间跳的是若干个已经缩完的花(缩过的花也是点,但是在维护pre" role="presentation">pre的时候,还是需要沿着这朵花暴跳,因为还需要维护每个点的匹配信息,只考虑一朵花的话没法维护所有点的信息)
所以在跳跃的过程中,暴力把所有访问到的节点和花的并查集全部合并到lca" role="presentation">lca上面,表示他们的花根是lca" role="presentation">lca。

感觉我写的很不清晰

总而言之,我们来总结一下带花树算法的流程

1.每次找一个未匹配的点出来增广
2.在增广过程中,如果相邻点是白点,或者是同一朵花中的节点,则直接跳过这个点
3.如果相邻点是一个未被匹配过的白点,证明找到了增广路,沿着原有的pre" role="presentation">pre和match" role="presentation">match路径,对这一次的匹配结果进行更新
4.如果相邻点是一个被匹配过的白点,那么把这个点的匹配点丢进队列中,尝试能否让这个点的匹配点找到另外一个点进行匹配,从而可以增广。
(以上步骤同匈牙利算法)
5.如果相邻点是一个被匹配过的黑点,证明此时出现了奇环,我们需要将这个环缩成一个黑点。具体的实现过程是:找到他们的最近花公共祖先,也就是他们的花根,同时,沿着当前这两个点一路到花根,将花上的所有节点全部染成黑点(因为一朵花都是黑点),将原来的白点丢进栈中。同时,修改花上所有点的pre" role="presentation">pre,此时,只剩下花根并不与花内的节点相匹配。

以下是UOJ79" role="presentation">UOJ79模板题的代码

#include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #include<cmath> #include<algorithm> #include<set> #include<map> #include<vector> #include<queue> using namespace std; #define ll long long #define RG register #define MAX 555 #define MAXL 255555 inline int read() { RG int x=0,t=1;RG char ch=getchar(); while((ch<'0'||ch>'9')&&ch!='-')ch=getchar(); if(ch=='-')t=-1,ch=getchar(); while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar(); return x*t; } struct Line{int v,next;}e[MAXL]; int h[MAX],cnt=1; inline void Add(int u,int v){e[cnt]=(Line){v,h[u]};h[u]=cnt++;} int match[MAX],pre[MAX],f[MAX],vis[MAX],tim,dfn[MAX]; int n,m,ans; int getf(int x){return x==f[x]?x:f[x]=getf(f[x]);} int lca(int u,int v) {++tim;u=getf(u);v=getf(v);while(dfn[u]!=tim){dfn[u]=tim;u=getf(pre[match[u]]);if(v)swap(u,v);}return u; } queue<int> Q; void Blossom(int x,int y,int w) {while(getf(x)!=w){pre[x]=y,y=match[x];if(vis[y]==2)vis[y]=1,Q.push(y);if(getf(x)==x)f[x]=w;if(getf(y)==y)f[y]=w;x=pre[y];} } bool Aug(int S) {for(int i=1;i<=n;++i)f[i]=i,vis[i]=pre[i]=0;while(!Q.empty())Q.pop();Q.push(S);vis[S]=1;while(!Q.empty()){int u=Q.front();Q.pop();for(int i=h[u];i;i=e[i].next){int v=e[i].v;if(getf(u)==getf(v)||vis[v]==2)continue;if(!vis[v]){vis[v]=2;pre[v]=u;if(!match[v]){for(int x=v,lst;x;x=lst)lst=match[pre[x]],match[x]=pre[x],match[pre[x]]=x;return true;}vis[match[v]]=1,Q.push(match[v]);}else{int w=lca(u,v);Blossom(u,v,w);Blossom(v,u,w);}}}return false; } int main() {n=read();m=read();for(int i=1;i<=m;++i){int u=read(),v=read();Add(u,v);Add(v,u);}for(int i=1;i<=n;++i)if(!match[i])ans+=Aug(i);printf("%dn",ans);for(int i=1;i<=n;++i)printf("%d ",match[i]);puts("");return 0; }

相关知识

图论 —— 带花树算法
【js学习笔记】“花密”算法本地化
从匈牙利算法到带权带花树——详解对偶问题在图匹配上的应用
利用带花树算法解决一般图的最大匹配
一般图最大匹配:带花树入门详解
鲜花分类算法
机器学习入门(一) 之 K近邻算法(KNN算法)
《统计学习方法》第 2 章“感知机”学习笔记
机器学习笔记(通俗易懂)
机器学习笔记——利用sklearn中KNN算法实现鸢尾花分类

网址: 带花树算法学习笔记 https://m.huajiangbk.com/newsview2211957.html

所属分类:花卉
上一篇: 4.2 主动运输与胞吞、胞吐
下一篇: 研学第一站——广元千佛崖