【題解】[USACO07OPEN]Dining G
阿新 • • 發佈:2020-08-10
\(\text{Solution:}\)
這一題,我們要做到,食物和牛、牛和飲料均為一對一的關係。我們發現這個圖不好建立。
經典技巧:將牛拆邊,拆成入點和出點,並連容量為\(1\)的邊。
然後,從源點向食物連邊,從食物向牛的入點連邊,入點向出點連邊,出點向飲料連邊,飲料向匯點連邊。容量均為\(1.\)
建立完這個圖模型後,直接跑最大流即可。
注意點的編號問題。
#include<bits/stdc++.h> using namespace std; const int MAXN=3e5+10; struct edge{ int nxt,to,flow; }e[MAXN]; int tot=1,head[MAXN],n,m; const int inf=2147483647; int cur[MAXN],F,D,S,T; int f[MAXN],d[MAXN],dep[MAXN]; inline void add(int x,int y,int w){ e[++tot].to=y; e[tot].nxt=head[x]; head[x]=tot;e[tot].flow=w; e[++tot].to=x;e[tot].nxt=head[y]; e[tot].flow=0;head[y]=tot; } bool bfs(int s,int t){ memset(dep,0,sizeof(dep)); queue<int>q;q.push(s); dep[s]=1;cur[s]=head[s]; for(;!q.empty();){ s=q.front();q.pop(); for(int i=head[s];i;i=e[i].nxt){ int j=e[i].to; if(!dep[j]&&e[i].flow>0){ dep[j]=dep[s]+1; cur[j]=head[j]; if(j==t)return true; q.push(j); } } } return false; } int dfs(int s,int flow,int t){ if(s==t||flow<=0)return flow; int rest=flow; for(int i=cur[s];i;i=e[i].nxt){ int j=e[i].to; if(e[i].flow>0&&dep[j]==dep[s]+1){ int tmp=dfs(j,min(rest,e[i].flow),t); if(tmp<=0)dep[j]=0; rest-=tmp;e[i].flow-=tmp;e[i^1].flow+=tmp; if(rest<=0)break; } } return flow-rest; } int dinic(int s,int t){ int ans=0; for(;bfs(s,t);)ans+=dfs(s,inf,t); return ans; } int main(){ scanf("%d%d%d",&n,&F,&D); S=0;T=100000; for(int i=1;i<=F;++i)add(S,i,1); for(int i=1;i<=D;++i)add(i+F+n,T,1); for(int i=1;i<=n;++i)add(i+F,i+D+F+n+1,1); for(int i=1;i<=n;++i){ scanf("%d%d",&f[i],&d[i]); for(int j=1;j<=f[i];++j){ int x;scanf("%d",&x); add(x,i+F,1); } for(int j=1;j<=d[i];++j){ int x;scanf("%d",&x); add(i+D+F+n+1,x+F+n,1); } } printf("%d\n",dinic(S,T)); return 0; }