如何建图?

  最开始的问题就是,怎么表示一只牛有了食物和饮料呢?

  后来发现可以先将食物与牛匹配,牛再去和饮料匹配,实际上这就构成了三个层次。

  起点到食物层边的容量是1,食物层到奶牛层容量是1,奶牛层到饮料层容量是1,饮料层到终点容量是1。

  但是后来发现有一组hack数据:

  2 3 3

  3 3 1 2 3 1 2 3
  3 3 1 2 3 1 2 3

  我们发现一头奶牛居然吃了多个套餐,所以要解决这个只需要将自己与自己建立一条容量是1的边就行了。

  

#include <cstdio>
#include <cstring>
#include <stack>
using namespace std;
#define sc scanf
#define pt printf
#define maxe 40960
#define maxv 405
#define maxn 1000
#define mll long long
const int inf = 0x3f3f3f3f;
int mn(int a,int b) { return a<b?a:b; }
int s,t, N,F,D;
typedef struct ed{
int v,w,cap,flow;
} ed;
ed e[maxe];
int head[maxv],nxt[maxe],tot,dis[maxv];
void init()
{
tot = ;
memset(head,-,sizeof(head));
} void add(int u,int v,int cap,int flow)
{
e[tot].v=v;
e[tot].cap=cap;
e[tot].flow=flow;
nxt[tot]=head[u];
head[u]=tot++; e[tot].v=u;
e[tot].cap=flow;
e[tot].flow=;
nxt[tot]=head[v];
head[v]=tot++;
}
int dfs(int u,int exp)
{
if(exp==||u==t) return exp;
int i,v,flow=,tmp;
for(i=head[u];i!=-;i=nxt[i])
{
v=e[i].v;
//pt("u=%d,v=%d\n",u,v); if(dis[v]==dis[u]+)
{
tmp = dfs(v,mn(e[i].cap-e[i].flow,exp));
if(tmp==) continue;
// pt("u=%d,v=%d,tmp=%d\n",u,v,tmp);
e[i].flow += tmp;
e[i^].flow -= tmp; exp-=tmp;
flow+=tmp; if(exp==) break;
}
}
//pt("wt\n");
if(flow==) dis[u]=inf;
return flow;
}
stack<int> q;
int main()
{
freopen("in.txt","r",stdin);
while(~sc("%d%d%d",&N,&F,&D))
{
//pt("OK\n");
int i,j,ans=,u,v,tt,FF,DD;
init(); s=,t=+*N+F+D;
// pt("OK\n");
// F使用1-F N使用(F+1 - F+N)(F+1 + N - F+ 2*N) D使用 F+2*N+1 - F+2*N+D
for(i=;i<=F;++i) add(,i,,);
for(i=F+*N+;i<=F+*N+D;++i) add(i,t,,);
for(i=F+;i<=F+N;++i) add(i,i+N,,);
//pt("OK\n");
for(i=;i<=N;++i)
{
sc("%d%d",&FF,&DD);
for(j=;j<=FF;++j)
{
sc("%d",&tt);
add(tt,F+i,,);
}
for(j=;j<=DD;++j)
{
sc("%d",&tt);
add(F+N+i,F+*N+tt,,);
}
}
//pt("OK\n");
while()
{
//BFS建立层次图
memset(dis,inf,sizeof(dis));
dis[s]=;
while(!q.empty()) q.pop();
q.push(s);
while(!q.empty())
{ u=q.top(); q.pop();
for(i=head[u];i!=-;i=nxt[i])
{
if(e[i].cap - e[i].flow <= ) continue;
v = e[i].v; if(dis[u]+<dis[v])
{
dis[v] = dis[u] +;
//pt("BFS: u=%d,v=%d\n",u,v);
if(v==t) break;
q.push(v);
} }
}
if(dis[t]==inf) break;
//DFS进行增广
tt=dfs(,inf);
if(tt==) break;
else ans+=tt;
//pt("tt=%d\n",tt); }
pt("%d\n",ans);
}
return ;
}

POJ 3281

POJ 3281 网络流 拆点保证本身只匹配一对食物和饮料的更多相关文章

  1. poj 3281(网络流+拆点)

    题目链接:http://poj.org/problem?id=3281 思路:设一个超级源点和一个超级汇点,源点与食物相连,饮料与汇点相连,然后就是对牛进行拆点,一边喜欢的食物相连,一边与喜欢的饮料相 ...

  2. POJ 3281 网络流 拆点 Dining

    题意: 有F种食物和D种饮料,每头牛有各自喜欢的食物和饮料,而且每种食物或者饮料只能给一头牛. 求最多能有多少头牛能同时得到它喜欢的食物或者饮料. 分析: 把每个牛拆点,中间连一条容量为1的边,保证一 ...

  3. POJ 3281 网络流dinic算法

    B - Dining Time Limit:2000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Submit S ...

  4. POJ 3281 网络流

    题意: 思路: 网络流 重在建图- 建完了图 就一切都好说了 这道题 我的想法是 先把源点和所有的食品连上边 (容量为1) 再把食品和对应的奶牛连上边 (容量为1) 这个时候要拆点 因为每只奶牛只能才 ...

  5. ACM Computer Factory POJ - 3436 网络流拆点+路径还原

    http://poj.org/problem?id=3436 每台电脑有$p$个组成部分,有$n$个工厂加工电脑. 每个工厂对于进入工厂的半成品的每个组成部分都有要求,由$p$个数字描述,0代表这个部 ...

  6. POJ 3281 Dining (拆点)【最大流】

    <题目链接> 题目大意: 有N头牛,F种食物,D种饮料,每一头牛都有自己喜欢的食物和饮料,且每一种食物和饮料都只有一份,让你分配这些食物和饮料,问最多能使多少头牛同时获得自己喜欢的食物和饮 ...

  7. poj 3281 Dining 拆点 最大流

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

  8. POJ 3281 [网络流dinic算法模板]

    题意: 农场主有f种食物,d种饮料,n头牛. 接下来的n行每行第一个数代表第i头牛喜欢吃的食物数量,和第i头牛喜欢喝的饮料数目. 接下来分别是喜欢的食物和饮料的编号. 求解:农场主最多能保证几头牛同时 ...

  9. B - Dining POJ - 3281 网络流

    Cows are such finicky eaters. Each cow has a preference for certain foods and drinks, and she will c ...

随机推荐

  1. Segment tree Beats

    Segment tree Beats Segment tree Beats,吉司机线段树,主要是关于如何用线段树实现区间取min/max.我们先看一道例题: HDU5306 Gorgeous Sequ ...

  2. java视频资源

    1.Java基础阶段 尚学堂_刘凯立_JavaSE基础视频 http://pan.baidu.com/s/1geCoY11 尚学堂_高淇_Java300集视频教程 https://pan.baidu. ...

  3. SCUT - 38 - 屠场的秘密 - 分解

    https://scut.online/p/38 要求是2016的倍数,把每个数分解成有2016的倍数和余数,两数余数的乘积是2016的倍数,则原数的乘积也是2016的倍数.

  4. jenkins使用记录转自https://my.oschina.net/sanpeterguo/blog/197931

    摘要: jenkins(持续集成开源工具)提供了丰富的api接口,基本上所有的操作都可以使用curl来从后台调度,包括:创建项目,禁用项目,启用项目,获取项目描述,获取配置文件,普通触发,scm触发, ...

  5. C#开发 WinForm如何在选项卡中集成加载多个窗体 实现窗体复用

    http://blog.csdn.net/upi2u/article/details/37914909 最近需要做的一个项目,为了避免从菜单中选择的麻烦,需要把几个窗体集成到一起,通过TabContr ...

  6. latex中\large的作用域问题

    在毕业论文的写作过程中,遇到了一个\large 作用域的问题.假设下面有三种写法: I am cool \large{you are right}, yeah, yeah, yeah I am coo ...

  7. JavaScript深入之从原型到原型链(转载)

    构造函数创建对象 我们先使用构造函数创建一个对象: function Person() { } var person = new Person(); person.name = 'Kevin'; co ...

  8. linux Sersync 上配置客户端

    1.安装 Rsync 并配置相关权限 在 SERSYNC 上配置 RSYNC 客户端相关权限认证: [root@SERSYNC /]# yum install rsync -y [root@SERSY ...

  9. V7双雄-基于Virtex7XC7VX690T的高性能计算板卡解决方案

    北京太速V7双雄-基于Virtex7XC7VX690T的高性能计算板卡

  10. Linux性能优化从入门到实战:15 文件系统篇:磁盘 I/O

    磁盘   磁盘是可以持久化存储的设备,按照存储介质来分类:   (1)机械磁盘(硬盘驱动器,Hard Disk Driver,HDD),主要由盘片和读写磁头组成,数据就存储在盘片的环状磁道中.在读写数 ...