Description
设有M个工人x1, x2, …, xm,和N项工作y1, y2, …, yn,规定每个工人至多做一项工作,而每项工作至多分配一名工人去做。由于种种原因,每个工人只能胜任其中的一项或几项工作。问应怎样分配才能使尽可能多的工人分配到他胜任的工作。这个问题称为人员分配问题。
Input
第一行两个整数m,n分别为工人数和工作数。
接下来一个整数s,为二分图的边数。
接下来s行,每行两个数ai,bi表示第ai个工人能胜任第bi份工作
Output
一个整数,表示最多能让多少个工人派到自己的胜任的工作上。
Sample Input
3 3
4
1 2
2 1
3 3
1 3
Sample Output
3
Hint
规模:
1<=m,n<=100
1<=s<=10000
思路:
这就是一个二分图匹配模板
#include<iostream>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<queue>
#include<stack>
#include<map>
#include<cstring>
using namespace std;
int n, m, s, tot, head[110];
int cover[110], link[110];
struct node{
int to, next;
}b[10010];
void add(int x, int y)
{
b[++tot]=(node){y, head[x]};
head[x]=tot;
}
bool find(int x)
{
for(int i=head[x]; i; i=b[i].next)
{
int y=b[i].to;
if(!cover[y])
{
cover[y]=1;
int q=link[y];
link[y]=x;
if(q==0||find(q))
return true;
link[y]=q;
}
}
return false;
}
int main(){
scanf("%d%d%d", &n, &m, &s);
for(int i=1; i<=s; i++)
{
int x, y;
scanf("%d%d", &x, &y);
add(x, y);
}
for(int i=1; i<=n; i++)
{
memset(cover, 0, sizeof(cover));
find(i);
}
int ans=0;
for(int i=1; i<=m; i++)
if(link[i]!=0)
ans++;
printf("%d", ans);
return 0;
}