<题目链接>

题目大意:

有N头牛,F种食物,D种饮料,每一头牛都有自己喜欢的食物和饮料,且每一种食物和饮料都只有一份,让你分配这些食物和饮料,问最多能使多少头牛同时获得自己喜欢的食物和饮料。

解题分析:

开始还以为是一道匹配问题,后面才知道这是用网络流求解。

首先我们要明确,如果按照源点——>食物——>牛——>饮料——>汇点这样建图,是不符合题目条件的。因为题目要求每头牛只能吃一份食物和饮料,而这样建图,如果一头牛对应多个食物和饮料,那这样那头牛是可以吃多份食物和饮料的,所以我们需要对牛进行拆点,并且拆除的两点之间用容量为1边相连,这样跑最大流的时候就能够限制每头牛最多只能吃一份食物和饮料了。下面是Dinic的模板。

 #include <cstdio>
 #include <cstring>
 #include <queue>
 #include <algorithm>
 using namespace std;

 #define INF 0x3f3f3f3f
 ;
 const int maxm = 1e5; 

 int depth[maxn], vis[maxn],cur[maxn], head[maxn];
 int N, F, D;
 int sect,cnt;    //sect为汇点
 struct Edge {
     int v, cap, flow, next;
 }edge[maxm];

 void init(){
     cnt = ;
     memset(head, -, sizeof(head));
 }

 void add(int u, int v, int w){
     edge[cnt].v = v, edge[cnt].cap = w, edge[cnt].flow = ,edge[cnt].next = head[u]; //正向弧(容量为w)
     head[u] = cnt++;
     edge[cnt].v = u, edge[cnt].cap = , edge[cnt].flow = ,edge[cnt].next = head[v]; //反向弧(容量为0)
     head[v] = cnt++;
 }

 void getmap(){
     int tmp1, tmp2;
     ; i <= N; ++i){
         scanf("%d%d", &tmp1, &tmp2);
         while(tmp1--){
             int num;
             scanf("%d", &num);
             add( * N + num, i, );//食物和左牛连接
         }
         while(tmp2--){
             int num;
             scanf("%d", &num);
             add(N + i,  * N + F + num, );//右牛和饮料连接
         }
         add(i, i + N, );//左牛和右牛连接
     }
     sect =  * N + F + D + ;
     ; i <= F; ++i)
         add(,  * N + i, );//源点和食物连接
     ; i <= D; ++i)
         add( * N + F + i,  * N + F + D + , );//饮料和超级汇点连接
 }

 bool BFS(int st, int ed){    //构建分层图,并且判断增广路径是否存在
     queue<int>q;
     memset(depth, -, sizeof(depth));    //将所有点分层,初始化深度为-1
     memset(vis,  ,sizeof(vis));
     q.push(st);
     depth[st] = ;   //源点深度为0
     vis[st] = ;
     while(!q.empty()){
         int u = q.front();
         q.pop();
         ; i = edge[i].next){
             Edge &E = edge[i];
             if(!vis[E.v] && E.cap > E.flow){
                 vis[E.v] = ;
                 depth[E.v] = depth[u] + ;    //下一个点是当前点的深度+1
                 if(E.v == ed) return true;    //找到汇点则直接返回
                 q.push(E.v);
             }
         }
     }
     return false;    //没有找到通向汇点的增广路径
 }

 int DFS(int x, int ed, int val){
     )
         return val;
     , f;
     ; i = edge[i].next){  //注意这里的&符号,这样i增加的同时也能改变cur[u]的值,达到更新当前弧的目的
         Edge E=edge[i];
          && (f = DFS(E.v, ed, min(val, E.cap - E.flow))) > ){  //f为残余网络中增广路径的最小残量
             edge[i].flow += f;    //进行增广
             edge[i ^ ].flow -= f;   //正向流量+f,相当于反向流量-f
             flow += f;
             val -= f;
             ) break;
         }
     }
     return flow;
 }

 int Dinic(int st, int ed){
     ;  //最大流
     while(BFS(st, ed)){     //判断是否存在增广路
         memcpy(cur, head, sizeof(head));    //当前弧优化
         sumflow += DFS(st, ed, INF);
     }
     return sumflow;
 }

 int main (){
     while(scanf("%d%d%d", &N, &F, &D) != EOF){
         init();
         getmap();    //建图
         printf(, sect));
     }
     ;
 }

2018-11-23

POJ 3281 Dining (拆点)【最大流】的更多相关文章

  1. poj 3281 Dining 拆点 最大流

    题目链接 题意 有\(N\)头牛,\(F\)个食物和\(D\)个饮料.每头牛都有自己偏好的食物和饮料列表. 问该如何分配食物和饮料,使得尽量多的牛能够既获得自己喜欢的食物又获得自己喜欢的饮料. 建图 ...

  2. POJ 3281 Dining (网络流之最大流)

    题意:农夫为他的 N (1 ≤ N ≤ 100) 牛准备了 F (1 ≤ F ≤ 100)种食物和 D (1 ≤ D ≤ 100) 种饮料.每头牛都有各自喜欢的食物和饮料, 而每种食物或饮料只能分配给 ...

  3. POJ 3281 Dining(最大流)

    POJ 3281 Dining id=3281" target="_blank" style="">题目链接 题意:n个牛.每一个牛有一些喜欢的 ...

  4. POJ 3281 Dining (网络流)

    POJ 3281 Dining (网络流) Description Cows are such finicky eaters. Each cow has a preference for certai ...

  5. poj 3281 Dining 网络流-最大流-建图的题

    题意很简单:JOHN是一个农场主养了一些奶牛,神奇的是这些个奶牛有不同的品味,只喜欢吃某些食物,喝某些饮料,傻傻的John做了很多食物和饮料,但她不知道可以最多喂饱多少牛,(喂饱当然是有吃有喝才会饱) ...

  6. POJ 3281 网络流 拆点保证本身只匹配一对食物和饮料

    如何建图? 最开始的问题就是,怎么表示一只牛有了食物和饮料呢? 后来发现可以先将食物与牛匹配,牛再去和饮料匹配,实际上这就构成了三个层次. 起点到食物层边的容量是1,食物层到奶牛层容量是1,奶牛层到饮 ...

  7. POJ 3281 Dining(最大流+拆点)

    题目链接:http://poj.org/problem?id=3281 题目大意:农夫为他的 N (1 ≤ N ≤ 100) 牛准备了 F (1 ≤ F ≤ 100)种食物和 D (1 ≤ D ≤ 1 ...

  8. 图论--网络流--最大流--POJ 3281 Dining (超级源汇+限流建图+拆点建图)

    Description Cows are such finicky eaters. Each cow has a preference for certain foods and drinks, an ...

  9. POJ 3281 Dining(网络流拆点)

    [题目链接] http://poj.org/problem?id=3281 [题目大意] 给出一些食物,一些饮料,每头牛只喜欢一些种类的食物和饮料, 但是每头牛最多只能得到一种饮料和食物,问可以最多满 ...

随机推荐

  1. 【CSS】Bootstrap中select2+popover冲突

    网上搜索得到: It changes the position because the position is based on the popover's dimansions and select ...

  2. UEFI rootkit 工具LoJax可以感染电脑主板(mainboard)

    1.UEFI(Unified Extensible Firmware Interface)统一扩展接口,UEFI rootkit是以在UEFI中植入rootkit ,18年9月份ESET首次公开了境外 ...

  3. Metasploit框架问题

    1.使用nmap 扫描SMB服务漏洞信息 nmap -P0 --script=smb-check-vulns 10.10.15.123 2.sql语句绕过后台 这部分我待会补充 只有尽可能的过滤,没有 ...

  4. Best Free Hacking E-Books 2017 In PDF Format

    1.Best Free Hacking E-Books 2017 In PDF Format: 电子书籍下载地址 后续我会更新在我的百度云资源 上,需要的留言Black Belt Hacking &a ...

  5. 状态压缩dp小结

    最近一段时间算是学了一些状态压缩的题目,在这里做个小结吧 首先是炮兵布阵类题目,这类题目一开始给定一个矩形,要求在上面放置炮兵,如果在一格放了炮兵那么周围的某些格子就不能放炮兵,求最大能放置炮兵的数量 ...

  6. 沈阳润才教育CRM

    一.CRM初始 CRM,客户关系管理系统(Customer Relationship Management).企业用CRM技术来管理与客户之间的关系,以求提升企业成功的管理方式,其目的是协助企业管理销 ...

  7. 20165206 2017-2018-2 《Java程序设计》第二周学习总结

    20165205 2017-2018-2 <Java程序设计>第一周学习总结 教材学习内容总结 java语言共有8种基本数据类型,分别是boolean.byte.short.char.in ...

  8. 494. Target Sum

    You are given a list of non-negative integers, a1, a2, ..., an, and a target, S. Now you have 2 symb ...

  9. [转] babel入门基础

    背景 babel的官网说babel是下一代的js语法编译器,现在自己也在很多项目中使用了babel,可是自己对babel的认识呢,只停留在从google和别人项目中copy的配置代码上,内心感到很不安 ...

  10. MySQL 官方 Docker 镜像的使用

    首先是pull image,这里我拉取的是5.6.35: $ sudo docker pull mysql:5.6.35 拉下来以后大可以按照官方的说明无脑启动,但是外部无法访问,所以绑定端口: $ ...