## [$AcWing$ $861$. 二分图的最大匹配](https://www.acwing.com/problem/content/863/)
### 一、题目描述
给定一个二分图,其中左半部包含 $n_1$ 个点(编号 $1∼n_1$),右半部包含 $n_2$ 个点(编号 $1∼n_2$),二分图共包含 $m$ 条边。
数据保证任意一条边的两个端点都不可能在同一部分中。
请你求出二分图的最大匹配数。
> 二分图的匹配:给定一个二分图 $G$,在 $G$ 的一个子图 $M$ 中,$M$ 的边集 ${E}$ 中的任意两条边都不依附于同一个顶点,则称 $M$ 是一个匹配。
>
二分图的最大匹配:所有匹配中包含边数最多的一组匹配被称为二分图的最大匹配,其边数即为最大匹配数。
**输入格式**
第一行包含三个整数 $n_1$、 $n_2$ 和 $m$。
接下来 $m$ 行,每行包含两个整数 $u$ 和 $v$,表示左半部点集中的点 $u$ 和右半部点集中的点 $v$ 之间存在一条边。
**输出格式**
输出一个整数,表示二分图的最大匹配数。
**数据范围**
$1≤n1,n2≤500,1≤u≤n_1,1≤v≤n_2$,$1≤m≤10^5$
**输入样例:**
```cpp {.line-numbers}
2 2 4
1 1
1 2
2 1
2 2
```
**输出样例:**
```cpp {.line-numbers}
2
```
### 二、二分图的最大匹配
* 前提:给定一个二分图,不用你判断是不是二分图,已经明确知道是二分图,让你求它的最大匹配。
* 定义:在一个无向图中,定义一条边覆盖的点为这条边的两个端点。
找到一个边集$S$包含最多的边,使得这个边集覆盖到的所有顶点中的每个顶点只被一条边覆盖。$S$的大小叫做图的最大匹配。
#### 匈牙利算法
又名:【**月佬算法**】,目标:成就更多美满姻缘!
建立有向图$G$,分为二分图的左侧和右侧。 优先选择左侧序号更小的连接可能的边。 对于两个点的目标点“**冲突**”的时候,采取“**协商**”的办法。 即序号小的连接可能连接的另一条边。 若“**协商**”失败,则放弃序号较大的点的边。
概念看的都好绕,实例比较好懂:
你拥有的大概就是下面这样一张关系图,每一条连线都表示互有好感。
本着救人一命,胜造七级浮屠的原则,你想要尽可能地撮合更多的情侣,匈牙利算法的工作模式会教你这样做:
1. 先试着给$1$号男生找妹子,发现第一个和他相连的$1$号女生还名花无主,$got\ it$,连上一条蓝线
2. 接着给$2$号男生找妹子,发现第一个和他相连的$2$号女生名花无主,$got\ it$
3. 接下来是$3$号男生,很遗憾$1$号女生已经有主了,怎么办呢?
我们试着给之前$1$号女生匹配的男生(也就是$1$号男生)另外分配一个妹子。(黄色表示这条边被临时拆掉)
与$1$号男生相连的第二个女生是$2$号女生,但是$2$号女生也有主了,怎么办呢?我们再试着给$2$号女生的原配重新找个妹子(注意这个步骤和上面是一样的,这是一个递归的过程)
此时发现$2$号男生还能找到$3$号女生,那么之前的问题迎刃而解了,回溯回去
| $2$号男生可以找$3$号妹子 | $1$号男生可以找$2$号妹子 | $3$号男生可以找$1$号妹子 |
| ---------------------------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------- |
|
|
|
|
所以第3步最后的结果就是:
4. 接下来是$4$号男生,很遗憾,按照第三步的节奏我们没法给$4$号男生腾出来一个妹子,我们实在是无能为力了……香吉士同学走好。
### 三、实现代码
```cpp {.line-numbers}
#include
using namespace std;
const int N = 510;
const int M = 100010;
int n1, n2;
int m;
int match[N];
int st[N];
int res;
int h[N], e[M], ne[M], idx;
void add(int a, int b) {
e[idx] = b, ne[idx] = h[a], h[a] = idx++;
}
int dfs(int u) {
for (int i = h[u]; ~i; i = ne[i]) {
int v = e[i];
if (!st[v]) {
st[v] = 1;
if (match[v] == 0 || dfs(match[v])) {
match[v] = u;
return 1;
}
}
}
return 0;
}
int main() {
memset(h, -1, sizeof h);
cin >> n1 >> n2 >> m;
while (m--) {
int a, b;
cin >> a >> b;
add(a, b);
}
for (int i = 1; i <= n1; i++) {
memset(st, 0, sizeof st);
if (dfs(i)) res++;
}
printf("%d\n", res);
return 0;
}
```
### 五、 解释`int st[N]`的作用
**重点**:`st[]`的作用:
首先我们$find(x)$ 遍历属于$h[x]$的单链表相当于遍历他所有喜欢的女生
如果某个女生$j$没有被他预定过的话,就标记这个女生$j$被他预定,即$st[j]=true$
这时如果女$j$还没有匹配过,即$match[j]==0$的时候,那这个预定就成真了,得到$match[j]=x$
而如果女$j$之前就被男$k$匹配过了,那我们就$find(k)$,也就是$find(match[j])$(因为原本$match[j]==k$)
然后在$find(k)$的过程中,因为$st[j]=true$,这时候男$k$就不能再选则女$j$了,因为女$j$已经被预定了,所以男$k$就只能在他喜欢的女生里面选择其他人来匹配。
当然,如果$find(k)$返回$false$的话,那就 $if(match[j]==0 || find(match[j]))$都不成立,那男$j$就一边玩去吧~