<题目链接>

题目大意:

有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. Confluence 6 重构查找索引

    查找索引是自动维护的,但是你有时候可能会因为你在查找的时候或查看者邮件主题出现了异常,或者你的 Confluence 实例升级到了新的版本,你可能需要手动重构索引. 进行搜索索引重构: 在屏幕的右上角 ...

  2. Confluence 6 配置一个数据源连接

    这个指南指导你如何配置使用 JNDI 数据源来连接到你的数据库.使用这个类型的连接,Confluence 将会询问应用服务器(Tomcat)中你配置的连接信息. 如果你希望使用 JDBC 的数据库连接 ...

  3. 1709: Fire or Retreat(zzuli)

    水题,哎,可是第一次是因为编译错了,vs不知咋了,无奈: 后面几次又因为类型用了int错了,痛苦: 题目描述 在与科技水平远胜于我们的外星人的战斗最后,我们能够用来对外星装甲造成伤害的武器只剩下了…… ...

  4. (不断更新)关于显著性检测的调研-Salient Object Detection: A Survey

    <Salient Object Detection: A Survey>作者:Ali Borji.Ming-Ming Cheng.Huaizu Jiang and Jia Li 基本按照文 ...

  5. LeetCode(101):对称二叉树

    Easy! 题目描述: 给定一个二叉树,检查它是否是镜像对称的. 例如,二叉树 [1,2,2,3,4,4,3] 是对称的. 1 / \ 2 2 / \ / \ 3 4 4 3 但是下面这个 [1,2, ...

  6. numpy 与 pandas

    numpy: import numpy as np np.array([1,2,3]) 创建数组 np.arange(10).reshape(2,5) 类似于range(起始,终止,步长),可以加re ...

  7. Python Day-1 练习

    作业1 要求:1.输入用户密码 2.认真成功后显示欢迎信息 3.输入三次锁定 代码如下: __author__ = 'zhang.ning' username = "zhangning&qu ...

  8. 删除Apache服务的命令

    转到\Apache24\bin目录下,使用cmd命令sc delete apache2.2

  9. 一脸懵逼学习Hadoop中的序列化机制——流量求和统计MapReduce的程序开发案例——流量求和统计排序

    一:序列化概念 序列化(Serialization)是指把结构化对象转化为字节流.反序列化(Deserialization)是序列化的逆过程.即把字节流转回结构化对象.Java序列化(java.io. ...

  10. Lambda表达式树构建(上)

    概述 Lambda是C#常用的语句,采用委托等方式,来封装真实的代码块.Lambda其实就是语法糖,是一个匿名函数,是一种高效的类似于函数式编程的表达式,Lambda简化了开发中需要编写的代码量.它可 ...