题目背景
二分图
题目描述
给定一个二分图,结点个数分别为n,m,边数为e,求二分图最大匹配数
输入输出格式
输入格式:第一行,n,m,e
第二至e+1行,每行两个正整数u,v,表示u,v有一条连边
输出格式:共一行,二分图最大匹配
输入输出样例
1 1 1
1 1
1
说明
n,m≤1000 , 1≤u≤n , 1≤v≤m
因为数据有坑,可能会遇到 v>mv>mv>m 的情况。请把 v>mv>mv>m 的数据自觉过滤掉。
算法:二分图匹配
Solution:
二分图最大匹配的模板,直接套上匈牙利算法,至于实现可以去看博客(因为别人写的太好了,感觉自己写还不如别人清楚,所以就不写二分图匹配的详解了)。
1 #include <iostream> 2 #include <cstdio> 3 #include <algorithm> 4 #include <cstring> 5 using namespace std; 6 const int N=1005; 7 inline int gi() 8 { 9 char x=getchar();int a=0,f=1; 10 while(x<'0'||x>'9'){if(x=='-')f=-1;x=getchar();} 11 while(x>='0'&&x<='9'){a=a*10+x-'0';x=getchar();} 12 return a*f; 13 } 14 int n,m,s,u,v; 15 struct edge 16 { 17 int v,ne; 18 }e[N*N<<1]; 19 int h[N],cnt; 20 inline void ins(int u,int v){ 21 e[++cnt].v=v;e[cnt].ne=h[u];h[u]=cnt; 22 } 23 int le[N]; 24 bool vis[N]; 25 bool find(int u){ 26 for(int i=h[u];i;i=e[i].ne){ 27 int v=e[i].v; 28 if(!vis[v]){ 29 vis[v]=1; 30 if(!le[v]||find(le[v])){ 31 le[v]=u; 32 return 1; 33 } 34 } 35 } 36 return 0; 37 } 38 int ans; 39 int main(){ 40 n=gi();m=gi();int t=gi(); 41 for(int i=1;i<=t;i++){u=gi();v=gi();if(v>m||u>n)continue;ins(u,v);} 42 for(int i=1;i<=m;i++){ 43 memset(vis,0,sizeof(vis)); 44 if(find(i)) ans++; 45 } 46 printf("%d\n",ans); 47 return 0; 48 }