1. 程式人生 > 實用技巧 >【題解】[USACO07OPEN]Dining G

【題解】[USACO07OPEN]Dining G

\(Link\)

\(\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;
}