
题目链接:
http://codeforces.com/gym/100825
题目大意:
N(N<=600)个点,每个点有个名字Si,R(R<=200)个生产商在R个点上,F(F<=200)个工厂在F个点上,不会有一个点既有生产商又有工厂
有T(T<=1000)个公司,每个公司能够到达Ci个点,并且一个只能运输一个生产商的货物。比如生产商1给工厂1运输货物需要用到公司1,那么其余生产商就不能用公司1
一个工厂需要任意一个生产商供应货物,求最多能够给多少个工厂供应货物。
题目思路:
【最大流】
因为一个公司只能被一个生产商使用,所以将每个公司拆点为A入和A出两个点,A入到A出中间流量为1
A公司能够到达的所有点向A入连一条流量为1的边,A出到所能能到达的点连一条流量为1的边。
设置超级源S和超级汇T,超级源S到每一个生产商连一条流量为1的边,每个工厂到超级汇T连一条流量为1的边。
跑一遍最大流即可。
//
//by coolxxx
//#include<bits/stdc++.h>
#include<iostream>
#include<algorithm>
#include<string>
#include<iomanip>
#include<map>
#include<stack>
#include<queue>
#include<set>
#include<bitset>
#include<memory.h>
#include<time.h>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
//#include<stdbool.h>
#include<math.h>
#define min(a,b) ((a)<(b)?(a):(b))
#define max(a,b) ((a)>(b)?(a):(b))
#define abs(a) ((a)>0?(a):(-(a)))
#define lowbit(a) (a&(-a))
#define sqr(a) ((a)*(a))
#define swap(a,b) ((a)^=(b),(b)^=(a),(a)^=(b))
#define mem(a,b) memset(a,b,sizeof(a))
#define eps (1e-10)
#define J 10000
#define mod 1000000007
#define MAX 0x7f7f7f7f
#define PI 3.14159265358979323
#pragma comment(linker,"/STACK:1024000000,1024000000")
#define N 3004
#define M 5000004
using namespace std;
typedef long long LL;
double anss;
LL aans;
int cas,cass;
int n,m,lll,ans;
int S,T,nn,m1,m2;
int d[N],vd[N],last[N];
struct xxx
{
int next,to,q;
}a[M];
int ma[N][N];
string ss;
string s[N];
map<string,int>id;
void add(int x,int y,int z)
{
a[++lll].to=y;
a[lll].q=z;
a[lll].next=last[x];
last[x]=lll;
}
int sap(int u,int f)
{
int i,v,tt,asp=,mix=nn-;
if(u==T)return f;
for(i=last[u];i!=;i=a[i].next)
{
v=a[i].to;
if(a[i].q>)
{
if(d[u]==d[v]+)
{
tt=sap(v,min(f-asp,a[i].q));
asp+=tt;
a[i].q-=tt;
a[i^].q+=tt;
if(asp==f || d[S]==nn)
return asp;
}
mix=min(mix,d[v]);
}
}
if(asp!=)return asp;
if(!--vd[d[u]])d[S]=nn;
else vd[d[u]=mix+]++;
return asp;
}
int main()
{
#ifndef ONLINE_JUDGE
// freopen("1.txt","r",stdin);
// freopen("2.txt","w",stdout);
#endif
int i,j,k;
int x,y,z,f;
// init();
// for(scanf("%d",&cass);cass;cass--)
// for(scanf("%d",&cas),cass=1;cass<=cas;cass++)
// while(~scanf("%s",s))
while(~scanf("%d",&n))
{
lll=;
scanf("%d%d%d",&m1,&m2,&m);
S=n+m+m+,T=n+m+m+;
for(i=;i<=m1;i++)
{
cin>>ss;
if(id.find(ss)==id.end())
id[ss]=++cass;
j=id[ss];
add(S,j,);
add(j,S,);
}
for(i=;i<=m2;i++)
{
cin>>ss;
if(id.find(ss)==id.end())
id[ss]=++cass;
j=id[ss];
add(j,T,);
add(T,j,);
}
for(i=;i<=m;i++)
{
add(n+i+i-,n+i+i,),add(n+i+i,n+i+i-,);
scanf("%d",&cas);
for(j=;j<=cas;j++)
{
cin>>s[j];
if(id.find(s[j])==id.end())
id[s[j]]=++cass;
k=id[s[j]];
add(k,n+i+i-,),add(n+i+i-,k,);
add(n+i+i,k,),add(k,n+i+i,);
}
}
nn=T;
vd[]=nn;
while(d[S]<nn)
{
f=sap(S,MAX);
ans+=f;
}
printf("%d\n",ans); cass=;ans=;
mem(last,);mem(d,);mem(vd,);
id.clear();
}
return ;
}
/*
// //
*/