给定一个二分图,其中左半部包含n1n1个点(编号1~n1n1),右半部包含n2n2个点(编号1~n2n2),二分图共包含m条边。
数据保证任意一条边的两个端点都不可能在同一部分中。
请你求出二分图的最大匹配数。
二分图的匹配:给定一个二分图G,在G的一个子图M中,M的边集{E}中的任意两条边都不依附于同一个顶点,则称M是一个匹配。
二分图的最大匹配:所有匹配中包含边数最多的一组匹配被称为二分图的最大匹配,其边数即为最大匹配数。
输入格式
第一行包含三个整数 n1n1、 n2n2 和 mm。
接下来m行,每行包含两个整数u和v,表示左半部点集中的点u和右半部点集中的点v之间存在一条边。
输出格式
输出一个整数,表示二分图的最大匹配数。
数据范围
1≤n1,n2≤500
1≤u≤n1
1≤v≤n2
1≤m≤1e5
输入样例:
2 2 4
1 1
1 2
2 1
2 2
输出样例:
2
思路:二分图的最大匹配
#include<bits/stdc++.h>
#define mod 19260817
#define ll long long
#define inf 0x3f3f3f3f
using namespace std;
const int maxn = 1e5+10;
vector<int>q[510];
int book[510],cp[510];
//book标记n2中的元素是否被询问
//cp储存n2中的元素的匹配对象
bool can_find(int x)
{
int len=q[x].size();
for(int i = 0; i < len; i++)
{
int u=q[x][i];
//book防止重复询问
if(!book[u])//这个元素没有被询问过
{
book[u]=1;
//如果元素u没有匹配对象或者u的匹配对象可以找到一个新的匹配对象
//则让x与u进行匹配
if(!cp[u]||can_find(cp[u]))
{
cp[u]=x;
return true;
}
}
}
return false;
}
int main()
{
int n1,n2,m,a,b;
scanf("%d %d %d",&n1,&n2,&m);
for(int i = 1; i <= m; i++)
{
scanf("%d %d",&a,&b);
q[a].push_back(b);
}
int ans=0;
for(int i = 1; i <= n1; i++)
{
if(can_find(i))ans++;
}
printf("%d\n",ans);
return 0;
}