poj3281
非常非常经典的构图
有二分图学习基础的话,很容易想到这是一个“三分图”的匹配问题
我们将牛,food,drink作为点
为了方便,我们将牛放在中间,每头牛的出边指向drink种类,入边由food指入
建立超级源点指向所有food,超级汇点指向所有drink,
要满足最多的牛,也就是求一个最大流
但注意,如果这样求最大流的话,会经过牛点不止一次(因为牛会有多个入边和多个出边)
所以我们考虑将牛点拆为两个点,中间流量为1,这样就能保证牛只经过1次了
const max=;
var a:array[..,..] of longint;
numh,h,cur,pre:array[..] of longint;
n,t,i,j,m,s,f,d,ans,x:longint; procedure sap;
var i,j,flow,tmp,neck,u,k:longint;
begin
numh[]:=;
u:=;
while h[]<t+ do
begin
if u=t then
begin
i:=;
j:=cur[];
flow:=max;
while i<>t do //其实这个地方多余了,容易知道,瓶颈边的流量一定为1
begin
if flow>a[i,j] then
begin
neck:=i;
flow:=a[i,j];
end;
i:=j;
j:=cur[j];
end;
inc(ans,flow);
i:=;
j:=cur[i];
while i<>t do
begin
dec(a[i,j],flow);
inc(a[j,i],flow);
i:=j;
j:=cur[i];
end;
u:=neck;
end;
k:=-;
for i:= to t do
if (a[u,i]>) and (h[u]=h[i]+) then
begin
k:=i;
break;
end;
if k<>- then
begin
cur[u]:=k;
pre[k]:=u;
u:=k;
end
else begin
dec(numh[h[u]]);
if numh[h[u]]= then break; //GAP优化
tmp:=t+;
for i:= to t do
if (a[u,i]>) then tmp:=min(tmp,h[i]); //更新标号
h[u]:=tmp+;
inc(numh[h[u]]);
if u<> then u:=pre[u];
end;
end;
end; begin
readln(n,m,s);
fillchar(a,sizeof(a),);
t:=*n+m+s+; //计算建图后总点数
for i:= to m do
a[,*n+i]:=;
for i:= to s do
a[*n+m+i,t]:=;
for i:= to n do
a[i,i+n]:=;
for i:= to n do
begin
read(f,d);
for j:= to f do
begin
read(x);
a[*n+x,i]:=;
end;
for j:= to d do
begin
read(x);
a[i+n,*n+m+x]:=;
end;
end;
fillchar(cur,sizeof(cur),);
fillchar(pre,sizeof(pre),);
fillchar(h,sizeof(h),);
fillchar(numh,sizeof(numh),);
sap;
writeln(ans);
end.
这题带给我们两个启示:
拆点和建立超级源汇点是网络流构图的基础而又重要的部分
网络流的建图比较复杂(这题还算简单),要细心检查……;
poj3281的更多相关文章
- POJ3281 Dining —— 最大流 + 拆点
题目链接:https://vjudge.net/problem/POJ-3281 Dining Time Limit: 2000MS Memory Limit: 65536K Total Subm ...
- Dining(POJ-3281)【最大流】
题目链接:https://vjudge.net/problem/POJ-3281 题意:厨师做了F种菜各一份,D种饮料各一份,另有N头奶牛,每只奶牛只吃特定的菜和饮料,问该厨师最多能满足多少头奶牛? ...
- POJ-3281(最大流+EK算法)
Dining POJ-3281 这道题目其实也是网络流中求解最大流的一道模板题. 只要建模出来以后直接套用模板就行了.这里的建模还需要考虑题目的要求:一种食物只能给一只牛. 所以这里可以将牛拆成两个点 ...
- poj3281 Dining
Dining Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 14316 Accepted: 6491 Descripti ...
- POJ3281 Dining(拆点构图 + 最大流)
题目链接 题意:有F种食物,D种饮料N头奶牛,只能吃某种食物和饮料(而且只能吃特定的一份) 一种食物被一头牛吃了之后,其余牛就不能吃了第一行有N,F,D三个整数接着2-N+1行代表第i头牛,前面两个整 ...
- POJ3281 Dining 最大流
题意:有f种菜,d种饮品,每个牛有喜欢的一些菜和饮品,每种菜只能被选一次,饮品一样,问最多能使多少头牛享受自己喜欢的饮品和菜 分析:建边的时候,把牛拆成两个点,出和入 1,源点向每种菜流量为1 2,每 ...
- poj3281(最大流)
传送门:Dining 题意:一些牛,一些食物,一些饮料,每头牛都有其喜欢的几种食物和几种饮料,求最多能给多少头牛即找到食物又找到饮料~也就是有多少个 牛---食物---饮料 的匹配,而且满足一一匹配, ...
- poj-3281(拆点+最大流)
题意:有n头牛,f种食物,d种饮料,每头牛有自己喜欢的食物和饮料,问你最多能够几头牛搭配好,每种食物或者饮料只能一头牛享用: 解题思路:把牛拆点,因为流过牛的流量是由限制的,只能为1,然后,食物和牛的 ...
- poj3281构图题
题目大意:有F种食物,D种饮料N头奶牛,只能吃某种食物和饮料(而且只能吃特定的一份)一种食物被一头牛吃了之后,其余牛就不能吃了第一行有N,F,D三个整数接着2-N+1行代表第i头牛,前面两个整数是Fi ...
随机推荐
- 两个list 求交集效率对比
__author__ = 'daitr' #--coding:utf-8-- import datetime #方法一: #a=[2,3,4,5] #b=[2,5,8] #tmp = [val for ...
- linux系统文件属性及企业精典故障案例
linux系统文件属性: [root@nginx_back ~]# stat keepalived-1.2.7.tar.gz 查看文件属性 File: "keepalived-1.2.7.t ...
- Educational Codeforces Round 12 E. Beautiful Subarrays 预处理+二叉树优化
链接:http://codeforces.com/contest/665/problem/E 题意:求规模为1e6数组中,连续子串xor值大于等于k值的子串数: 思路:xor为和模2的性质,所以先预处 ...
- 苹果Mac操作系统下怎么显示隐藏文件
对于新手而已民,苹果的MAC操作系统刚用时用得很不习惯,比如想要显示被隐藏的文件时,不像windows有个“文件夹选项”对话框可以来设置,百度出来的结果都是用命令来操作,但我建议不要用命令去操作, ...
- Comet、SSE、Web Socket
来自<javascript高级程序设计 第三版:作者Nicholas C. Zakas>的学习笔记(十一) Comet Comet是一种更加高级的Ajax技术("服务器推送&qu ...
- asp 文件上传(无组件上传)
文件1.上传界面文件 upload.htm<html><head><meta http-equiv="Content-Language" conten ...
- spoj 345
DP 想了好久 还是看了一下题解.... f[i][j]表示i到j全部合并后的最小花费,f[i][j] = min{f[i][k]+f[k+1][j]+d[i][k]*d[k+1][j]} (i ...
- uva 10404
dp 1表示先手赢 #include <iostream> #include <cstdio> #include <cstring> #include < ...
- CSS文件和Javascript文件的压缩
像JQuery一样来压缩我们的CSS和JS 我们都知道一般JQuery新版本发布的时候往往会有几个不同类型文件,比如原始版本文件.最小文件以及其他配合IDE智能提示的各种版本文件,前期我们使用JQue ...
- linux mysql添加用户
格式:grant select on 数据库.* to 用户名@登录主机 identified by "密码" 例1.增加一个用户user_1密码为123,让他可以在任何主机上登录 ...