如果只有食物或者饮料那就是个二分图最大匹配。

三个真想不出来。。然后看题解。。从源点到食物到牛到饮料到汇点,这样建图。

所以思维不能太局限了,不懂得把食物和饮料放到牛两边,以为牛吃食物饮料、食物饮料被牛吃是符合逻辑的=。=

 #include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
#define INF (1<<30)
#define MAXN 444
#define MAXM 444*444*2 struct Edge{
int v,cap,flow,next;
}edge[MAXM];
int vs,vt,NE,NV;
int head[MAXN]; void addEdge(int u,int v,int cap){
edge[NE].v=v; edge[NE].cap=cap; edge[NE].flow=;
edge[NE].next=head[u]; head[u]=NE++;
edge[NE].v=u; edge[NE].cap=; edge[NE].flow=;
edge[NE].next=head[v]; head[v]=NE++;
} int level[MAXN];
int gap[MAXN];
void bfs(){
memset(level,-,sizeof(level));
memset(gap,,sizeof(gap));
level[vt]=;
gap[level[vt]]++;
queue<int> que;
que.push(vt);
while(!que.empty()){
int u=que.front(); que.pop();
for(int i=head[u]; i!=-; i=edge[i].next){
int v=edge[i].v;
if(level[v]!=-) continue;
level[v]=level[u]+;
gap[level[v]]++;
que.push(v);
}
}
} int pre[MAXN];
int cur[MAXN];
int ISAP(){
bfs();
memset(pre,-,sizeof(pre));
memcpy(cur,head,sizeof(head));
int u=pre[vs]=vs,flow=,aug=INF;
gap[]=NV;
while(level[vs]<NV){
bool flag=false;
for(int &i=cur[u]; i!=-; i=edge[i].next){
int v=edge[i].v;
if(edge[i].cap!=edge[i].flow && level[u]==level[v]+){
flag=true;
pre[v]=u;
u=v;
//aug=(aug==-1?edge[i].cap:min(aug,edge[i].cap));
aug=min(aug,edge[i].cap-edge[i].flow);
if(v==vt){
flow+=aug;
for(u=pre[v]; v!=vs; v=u,u=pre[u]){
edge[cur[u]].flow+=aug;
edge[cur[u]^].flow-=aug;
}
//aug=-1;
aug=INF;
}
break;
}
}
if(flag) continue;
int minlevel=NV;
for(int i=head[u]; i!=-; i=edge[i].next){
int v=edge[i].v;
if(edge[i].cap!=edge[i].flow && level[v]<minlevel){
minlevel=level[v];
cur[u]=i;
}
}
if(--gap[level[u]]==) break;
level[u]=minlevel+;
gap[level[u]]++;
u=pre[u];
}
return flow;
}
int main(){
int n,f,d,a,b,c;
scanf("%d%d%d",&n,&f,&d); vs=; vt=n+n+f+d+; NV=vt+; NE=;
memset(head,-,sizeof(head));
for(int i=; i<=f; ++i) addEdge(vs,i,);
for(int i=; i<=d; ++i) addEdge(i+f+n+n,vt,);
for(int i=; i<=n; ++i){
addEdge(i+f,i+f+n,);
scanf("%d%d",&a,&b);
while(a--){
scanf("%d",&c);
addEdge(c,i+f,);
}
while(b--){
scanf("%d",&c);
addEdge(i+f+n,c+f+n+n,);
}
}
printf("%d",ISAP());
return ;
}

最新文章

  1. Hbuilder开发HTML5 APP之WebView
  2. 用Access作为后台数据库支撑,书写一个用C#写入记录的案例
  3. C#开发微信公众平台-就这么简单(附Demo)(转)
  4. 【CentOS】设置静态IP
  5. jeecms支持的freemaker标签大全
  6. VS2010界面主题更换全过程
  7. PHP htmlspecialchars() 函数
  8. js微博发布框
  9. SQL Server 2008 R2 的版本和组件
  10. poj 1753 Flip Game(bfs状态压缩 或 dfs枚举)
  11. 关于cgi、FastCGI、php-fpm、php-cgi
  12. POJ3041-Asteroids-匈牙利算法
  13. 【笔记】nodejs读取JSON,数组转树
  14. java web开发 高并发处理
  15. 编程菜鸟的日记-初学尝试编程-C++ Primer Plus 第4章编程练习7
  16. Java编程的逻辑 (91) - Lambda表达式
  17. 学习yii2.0——基础入门
  18. Asp.net core 学习笔记 ( ef core )
  19. PAT甲级 1121. Damn Single (25)
  20. [luogu2172] 部落战争

热门文章

  1. [Effective JavaScript 笔记]第21条:使用apply方法通过不同数量的参数调用函数
  2. Android 将可以按地点自动启动应用程序
  3. 2d背景循环
  4. rocksdb 编译安装 日志
  5. python操作Excel读写--使用xlrd
  6. explict关键字
  7. ReverseString
  8. 如何调试lua脚本
  9. cocos2d c++ 代码规范(译文)
  10. phpmyadmin 主机名自动补全