算法原理详见 http://www.csie.ntnu.edu.tw/~u91029/Matching.html

orz 带花树很神奇,挖坑最大权匹配

#include <iostream>
#include <cstring>
#include <cstdio>
#include <deque>
using namespace std;
const int maxn = ;
deque<int> p[maxn]; //树根到x的交错路径
bool adj[maxn][maxn];
int m[maxn]; //记录匹配情况
int d[maxn]; //记录奇点和偶点
int q[maxn], *qf, *qb; //记录可延伸的偶点
int n; void label_one_side(int x, int y, int bi){
for(int i = bi+; i < p[x].size(); i++){
int z = p[x][i];
if(d[z] == ){
p[z] = p[y];
p[z].insert(p[z].end(), p[x].rbegin(), p[x].rend()-i);
d[z] = ;
*qb++ = z;
}
}
} bool BFS(int r){
for(int i = ; i < n; i++) p[i].clear();
p[r].push_back(r);
for(int i = ; i < n; i++) d[i] = -;
d[r] = ;
qf = qb = q;
*qb++ = r;
while(qf < qb){
for(int x = *qf++, y = ; y < n; y++){
if(adj[x][y] && m[y] != y){
if(d[y] == -){
if(m[y] == -){
for(int i = ; i+ < p[x].size(); i += ){
m[p[x][i]] = p[x][i+];
m[p[x][i+]] = p[x][i];
}
m[x] = y; m[y] = x;
return true;
} else {
int z = m[y];
p[z] = p[x];
p[z].push_back(y);
p[z].push_back(z);
d[y] = ; d[z] = ;
*qb++ = z;
}
} else if(d[y] == ) {
int bi = ;
while(bi < p[x].size() && bi < p[y].size() && p[x][bi] == p[y][bi]) bi++;
bi--;
label_one_side(x, y, bi);
label_one_side(y, x, bi);
}
}
}
}
return false;
} int match(){
for(int i = ; i < n; i++) m[i] = -;
int c = ;
for(int i = ; i < n; i++){
if(m[i] == -)
if(BFS(i)) c++;
else m[i] = i;
}
return c;
} int main(){
cin>>n;
int x, y;
while(cin>>x>>y) adj[x][y] = adj[y][x] = true;
match();
}

这个是缩点版本的

#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
const int N = ;
int n;
bool adj[N][N];
int p[N];
int m[N];
int d[N];
int c1[N], c2[N];
int q[N], *qf, *qb; int pp[N];
int f(int x) { return x == pp[x] ? x : pp[x] = f(pp[x]); }
void u(int x, int y) { pp[x] = y; }
int v[N]; void path(int r, int x){
if(r == x) return;
if(d[x] == ){
path(r, p[p[x]]);
int i = p[x], j = p[p[x]];
m[i] = j; m[j] = i;
} else if(d[x] == ){
path(m[x], c1[x]);
path(r, c2[x]);
int i = c1[x], j = c2[x];
m[i] = j; m[j] = i;
}
} int lca(int x, int y, int r){
int i = f(x), j = f(y);
while(i != j && v[i] != && v[j] != ){
v[i] = ; v[j] = ;
if(i != r) i = f(p[i]);
if(j != r) j = f(p[j]);
}
int b = i, z = j; if(v[j] == ) swap(b, z);
for(int i = b; i != z; i = f(p[i])) v[i] = -;
v[z] = -;
return b;
} void contract_one_side(int x, int y, int b){
for(int i = f(x); i != b; i = f(p[i])){
u(i, b);
if(d[i] == ) c1[i] = x, c2[i] = y, *qb++ = i;
}
} bool BFS(int r){
for(int i = ; i < n; i++) pp[i] = i;
for(int i = ; i < n; i++) v[i] = d[i] = -;
d[r] = ;
qf = qb = q;
*qb++ = r;
while(qf < qb){
for(int x = *qf++, y = ; y < n; y++){
if(!adj[x][y] || m[y] == y || f(x) == f(y)) continue;
if(d[y] == -){
if(m[y] == -){
path(r, x);
m[x] = y; m[y] = x;
return true;
} else {
p[y] = x; p[m[y]] = y;
d[y] = ; d[m[y]] = ;
*qb++ = m[y];
}
} else if(d[f(y)] == ){
int b = lca(x, y, r);
contract_one_side(x, y, b);
contract_one_side(y, x, b);
}
}
}
return false;
} int main(){ }

Augmenting Path Algorithm : 一般图最大匹配的更多相关文章

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

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

  2. Maximum Cardinality Bipartite Matching: Augmenting Path Algorithm

    http://www.csie.ntnu.edu.tw/~u91029/Matching.html int nx,ny; int mx[N],my[N]; bool vy[N]; bool g[N][ ...

  3. [转载]Maximum Flow: Augmenting Path Algorithms Comparison

    https://www.topcoder.com/community/data-science/data-science-tutorials/maximum-flow-augmenting-path- ...

  4. UOJ79 一般图最大匹配

    题目描述 从前一个和谐的班级,所有人都是搞OI的.有 nn 个是男生,有 00 个是女生.男生编号分别为 1,-,n1,-,n. 现在老师想把他们分成若干个两人小组写动态仙人掌,一个人负责搬砖另一个人 ...

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

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

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

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

  7. 【UOJ#79】一般图最大匹配(带花树)

    [UOJ#79]一般图最大匹配(带花树) 题面 UOJ 题解 带花树模板题 关于带花树的详细内容 #include<iostream> #include<cstdio> #in ...

  8. ZOJ 3316 Game 一般图最大匹配带花树

    一般图最大匹配带花树: 建图后,计算最大匹配数. 假设有一个联通块不是完美匹配,先手就能够走那个没被匹配到的点.后手不论怎么走,都必定走到一个被匹配的点上.先手就能够顺着这个交错路走下去,最后一定是后 ...

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

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

随机推荐

  1. [原]解决phpstudy下的nginx无法运行的问题

    一直在用phpstudy下的apache,今天忽然想切换到nginx,出现了一些错误,最终还是解决了. 之前是php 5.3 + apache 现在是php 5.3n + nginx 问题就出在这n上 ...

  2. Git----使用WebHook实现代码自动部署

    起因: 经常本地push到gitee等线上代码仓库,然后登陆服务器在进行pull,很麻烦,想偷懒怎么办?使用git的webhook实现! 1.实现原理 1.1本地提交推送 1.2线上仓库监听push动 ...

  3. 在一台Apache服务器上创建多个站点(不同域名)

    使用不同的域名来区分不同的网站,所有的域名解析都指向同一个 IP 地址.Apache通过在HTTP头中附带的 host参数来判断用户需要访问哪一个网站. 例如要在一台服务器上设置如下两个站点: htt ...

  4. json_decode结果为null的几种原因

    值只能是UTF-8编码,元素最后不能有逗号,元素不能使用单引号,元素值中间不能有空格和n.

  5. 九、IIC驱动原理分析

    学习目标:学习IIC驱动原理: 一.IIC总线协议 IIC串行总线包括一条数据线(SDA)和一条时钟线(SCL),支持“一主多从”和“多主机”模式:每个从机设备都有唯一的地址来识别. 图 1 IIC ...

  6. 使用C6748和C5509A对nRF24L01驱动进行数据传输

    1. 写在前面 今天下午做了一个C5509A和C6748两个DSP的数据传输,经由RF24L01设备传输,都是模拟SPI协议,对于两个DSP来说,无非是配GPIO引脚,写好时序和延时.C5509A的G ...

  7. zabbix监控MySQL服务状态

    Mysql模板使用 在zabbix_agent配置文件中加入监控配置 vim etc/zabbix_agentd.conf ... UserParameter=mysql.version,mysqla ...

  8. Linux运维常用命令-linux服务器代维常用到的维护命令

    1.删除0字节文件find -type f -size 0 -exec rm -rf {} ; 2.查看进程按内存从大到小排列ps -e   -o "%C   : %p : %z : %a& ...

  9. MAVEN的项目升级

    今天我们来介绍一下版本依赖的问题 1.如果是admin的话,他要依赖于service的版本,则service的版本依赖于core的版本, 如果是本地编译,这我直接更新admin的就可以了,然后直接跑就 ...

  10. Ubuntu中搭建Hadoop集群(简记)

    stp1:在Vmware虚拟机上创建Ubantu.2环境 步骤:文件—>新建虚拟机—>典型(下一步)—>下一步——>位置(不建议放c盘,文件地址一定要全英文)—>下一步— ...