给定一个二分图,其中左半部包含 n1n1 个点(编号 1∼n11∼n1),右半部包含 n2n2 个点(编号 1∼n21∼n2),二分图共包含 mm 条边。
数据保证任意一条边的两个端点都不可能在同一部分中。
请你求出二分图的最大匹配数。
二分图的匹配:给定一个二分图 GG,在 GG 的一个子图 MM 中,MM 的边集 {E}{E} 中的任意两条边都不依附于同一个顶点,则称 MM 是一个匹配。
二分图的最大匹配:所有匹配中包含边数最多的一组匹配被称为二分图的最大匹配,其边数即为最大匹配数。
输入格式
第一行包含三个整数 n1n1、 n2n2 和 mm。
接下来 mm 行,每行包含两个整数 uu 和 vv,表示左半部点集中的点 uu 和右半部点集中的点 vv 之间存在一条边。
输出格式
输出一个整数,表示二分图的最大匹配数。
题解:
每个男生按顺序找女生 如果这个女生之前已经有男生,但之前这个男生还可以找到其他女生,则让他找别的女生,让现在这个男生找这个女生;
#include<bits/stdc++.h>
using namespace std;
const int N=510,M=100010;
int h[N],e[M],ne[M],idx;
int match[N];//女生对应的男生
int st[N];//记录每个女生是否被此次循环的男生找过
void add(int a,int b)
{
e[idx]=b;ne[idx]=h[a];h[a]=idx++;
}
bool find(int x)
{
for(int i=h[x];i!=-1;i=ne[i])//找每个与次男生相连的女生
{
int j=e[i];
if(!st[j])//如果女生没被找过
{
st[j]=true;//记录女生已经被找过了
if(match[j]==0||find(match[j]))//女生没有对应的男生或者女生之前对应的男生还可以找到
{
match[j]=x;//将次男生记录为这个女生对应的男生
return true;
}
}
}
return flase;//如果找不到 则查找失败
}
main()
{
int n1,n2,m;
cin>>n1>>n2>>m;
memset(h,-1,sizeof h);
while(m--)
{
int a,b;
cin>>a>>b;
add(a,b);
}
res=0;
for(int i=1;i<=n1;i++)//循环每个男生
{
memset(st,false,sizeof st);//每个男生开始找是先把女生初始化
if(find(i))res++;//如果找到了 就记一对
}
printf("%d",res);
return 0;
}