滚粗了的HansBug在收拾旧语文书,然而他发现了什么奇妙的东西。ios
蒟蒻HansBug在一本语文书里面发现了一本答案,然而他却明明记得这书应该还包含一份练习题。然而出如今他眼前的书多得数不胜数,其中有书,有答案,有练习册。已知一个完整的书册均应该包含且仅包含一本书、一本练习册和一份答案,然而如今全都乱作了一团。许多书上面的字迹都已经模糊了,然而HansBug仍是能够大体判断这是一本书仍是练习册或答案,而且可以大体知道一本书和答案以及一本书和练习册的对应关系(即仅仅知道某书和某答案、某书和某练习册有可能相对应,除此之外的均不可能对应)。既然如此,HansBug想知道在这样的状况下,最多可能同时组合成多少个完整的书册。网络
输入格式:spa
第一行包含三个正整数N一、N二、N3,分别表示书的个数、练习册的个数和答案的个数。rest
第二行包含一个正整数M1,表示书和练习册可能的对应关系个数。code
接下来M1行每行包含两个正整数x、y,表示第x本书和第y本练习册可能对应。(1<=x<=N1,1<=y<=N2)blog
第M1+3行包含一个正整数M2,表述书和答案可能的对应关系个数。get
接下来M2行每行包含两个正整数x、y,表示第x本书和第y本答案可能对应。(1<=x<=N1,1<=y<=N3)string
输出格式:it
输出包含一个正整数,表示最多可能组成完整书册的数目。io
样例说明:
如题,N1=5,N2=3,N3=4,表示书有5本、练习册有3本、答案有4本。
M1=5,表示书和练习册共有5个可能的对应关系,分别为:书4和练习册三、书2和练习册二、书5和练习册二、书5和练习册1以及书5和练习册3。
M2=5,表示数和答案共有5个可能的对应关系,分别为:书1和答案三、书3和答案一、书2和答案二、书3和答案3以及书4和答案3。
因此,以上状况的话最多能够同时配成两个书册,分别为:书2+练习册2+答案二、书4+练习册3+答案3。
数据规模:
对于数据点1, 2, 3,M1,M2<= 20
对于数据点4~10,M1,M2 <= 20000
相信大佬们一眼就能看出来这是个网络流最大流的模板题吧。
怎么作?直接把每个点都连起来跑最大流?
有没有注意到这个题每一本书只能用一次?这样的话要怎么办?
有人就会说了,这个好办,把全部的边的流量都赋成1不就好了吗?
额额,好像颇有道理的样子,可是对于这样一个图,
1 2 2
2
1 1
1 2
2
1 1
1 2
若是咱们对他跑最大流的话,咱们的结果会是2但实际应该是1,纵使咱们把全部的边权都赋成1最后的结果仍是2,那么咱们要怎么办呢?这个时候有人就会说了,拆点啊
咱们将中间的点拆成两个,而后是他们·直接的流量为1,这样保证每个点只被使用一次
#include<queue> #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> #define N 2000001 #define inf 9999999 using namespace std; queue<int>q; int s,e,x,y,m1,m2,ans,tot=1; int to[N],cap[N],cnt[N],lev[N],head[N],nextt[N]; int read() { int x=0,f=1; char ch=getchar(); while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar(); return x*f; } int add(int x,int y,int z) { tot++;to[tot]=y;cap[tot]=z;nextt[tot]=head[x];head[x]=tot; tot++;to[tot]=x;cap[tot]=0,nextt[tot]=head[y];head[y]=tot; } inline bool bfs() { while(!q.empty()) q.pop(); for(int i=s;i<=e;i++) { lev[i]=-1; cnt[i]=head[i]; } q.push(s),lev[s]=0; while(!q.empty()) { int x=q.front();q.pop(); for(int i=head[x];i;i=nextt[i]) { int t=to[i]; if(cap[i]>0&&lev[t]==-1) { lev[t]=lev[x]+1; q.push(t); if(t==e) return true; } } } return false; } int dinic(int x,int flow) { if(x==e) return flow; int delta,rest=0; for(int &i=cnt[x];i;i=nextt[i]) { int t=to[i]; if(cap[i]>0&&lev[t]>lev[x]) { delta=dinic(t,min(cap[i],flow-rest)); if(delta) { cap[i]-=delta; cap[i^1]+=delta; rest+=delta; if(rest==flow) break; } } } if(rest!=flow) lev[x]=-1; return rest; } int main() { int n1=read(),n2=read(),n3=read(); e=n1*2+n2+n3+1; m1=read(); for(int i=1;i<=m1;i++) { x=read(),y=read(); add(n1*2+y,x,1); } m2=read(); for(int i=1;i<=m2;i++) { x=read(),y=read(); add(x+n1,y+n1*2+n2,1); } for(int i=1;i<=n1;i++) add(i,n1+i,1); for(int i=1;i<=n2;i++) add(s,n1*2+i,1); for(int i=1;i<=n3;i++) add(n1*2+n2+i,e,1); while(bfs()) ans+=dinic(s,inf); printf("%d",ans); return 0; }