大意:给定v组cx与dx(表示喜欢cx不喜欢dx)或dx,cx问在v个人中最后还剩下多少人。
思路:既然是算最后剩下最多数目的人,那么我们可以对有矛盾的两个人建边(即结果是他们的满意度是最低的)。找到了这些人的个数sum,ans=v-sum/2,即匹配数。
#include<map>
#include<queue>
#include<cmath>
#include<cstdio>
#include<stack>
#include<iostream>
#include<cstring>
#include<algorithm>
#define LL int
#define inf 0x3f3f3f3f
#define eps 1e-8
#include<vector>
#define ls l,mid,rt<<1
#define rs mid+1,r,rt<<1|1
using namespace std;
char L[1000][5],H[1010][5];
int cro[1010],c,d,v,cnt;
bool vis[1010],mp[1000][1000];
int dfs(int u)
{
int i,j,k;
for(i=1;i <=v;i++)
{
if(!vis[i]&&mp[u][i])
{
vis[i]=true;
if(cro[i]==-1||dfs(cro[i]))
{
cro[i]=u;
return 1;
}
}
}
return 0;
}
int main(){
int n,m,i,j,k,cla;
scanf("%d",&cla);
while(cla--){
cnt = 0;
scanf("%d%d%d",&c,&d,&v);
memset(mp,false,sizeof(mp));
for(i = 0;i < v;++ i){
scanf("%s %s",L[i],H[i]);
}
for(i = 0;i < v;++ i){
for(j = 0;j < v;++ j){
if(strcmp(L[i],H[j])==0||strcmp(H[i],L[j])==0){
mp[j+1][i+1] = mp[i+1][j+1] = true;
}
}
}
int ans = 0;
memset(cro,-1,sizeof(cro));
for(i = 1;i <= v;++ i){
memset(vis,false,sizeof(vis));
if(dfs(i))
ans++;
}
printf("%d\n",ans);
}
return 0;
}