bzoj 2324 ZJOI 营救皮卡丘 费用流
题的大概意思就是给定一个无向图,边有权值,现在你有k个人在0点,要求走到n点,且满足
1:人们可以分头行动,可以停在某一点不走了
2:当你走到x时,前x-1个点必须全部走过(不同的人走过也行,即分两路走,其中一路走过1-x-1点
,另一路可以走x点)
3:道路可以多次走,点在满足2的情况下可以多次经过
求所有人走过的总路程最短
这道题历尽挫折。。。
首先我们知道想要走到x点,一定是从前x-1个点走过来,假设我们从y走过来,y<x,那么y走到x一定走的
是可以走的最短路,既然可以走x了,那么前x-1个点一定已经走过了,所以y到x的最短路就是当图只有前x点
时的y到x的最短路,那么我们可以用floyd解决,设d[k,i,j]代表只走前K个点,I到J的最短路,那么循环时,最
外层是k,直接记录d[k,i,j]就行了,因为当循环到K时,只用前K个更新过I到J的最短路,所以I到J只会走前K个点
,这时的d[i,j]就是d[k,i,j]了
那么我们对于这道题拆点,每个点拆成Ai,Bi,添加源S,汇T
规定(a,b,c,d)为连接a到b,容量为c,费用为d的有向边
那么我们先(Ai,Bi,1,0)代表这一个点被经过一次,因为每个点有效经过只有一次,下文具体解释
然后(S,0,k,0),代表从0开始走,一共可以走k个人
然后(Bi,T,1,0),代表一个人最终停在点I,就相当于这个流没用了,直接流出
然后我们对于每个点I,肯定会从之前的一个点走过来,这就是开始说的floyd处理的过程了,每个J,0<=j<i连向I
(J,I,1,d[I,J,I])
那么这么建的正确性到现在还没有保证,因为题中要求每个点都要被走过,这里只是连边时费用应用了下这个性质,但是
做费用流的时候是没法保证的(就是可能有的点不会被流过),那么Ai连Bi代表这个点被流过,那么我们规定必须流这个边就行了,
也就是经常说的流量的上下界(上界就是容量),规定Ai到Bi的上下界流量都为1
那么具体怎么做,假设对于点A到B规定上界为U(p),下界为D(own),那么我们一定会至少流U的,所以剩下的就是D-U,我们
将这条边的流量设为D-U,可这样流量平衡等式就不满足了,也就是A点应该多流出D的,B应该多流入D的,所以添加新的源点X,
汇点Y,(A,x,D,0),(y,B,D,0),(x,y,INF,0)也就是说我们把规定必须流的从X引走然后又流到Y了,这样就行了
对于下界还有一个二分的做法,内个不会。。。
然后这道题可以不用这么麻烦,可以将Ai到Bi的边的费用设成-INF,那么因为我们是最短路找增广路的,所以一定会走这一条边,
然后就好些很多了
对于这道题,还有另一种建图的方式,也是先拆点
然后没有0这个点
加入X,Y点
(s,Y,1,0)代表从0开始走,(Y,X,K,0)代表最多走K个人,
然后拆的点(Ai,Bi,-inf,0),(Bi,Aj,1,d[j,i,j])和上一种建图代表的一样,然后我们Y,X起的是0的作用
所以(X,Ai,1,d[i,0,i]]),然后再(Ai,T,1,0),代表走到I不走了
但是光这样建,如果K比N小的时候只会流K个,有的点流不到(上一种方法用的下界解决的),那么这时
(S,Bi,1,0)代表从0,走的人,之后去哪儿,这样直接求就行了。
还有地方可以优化,就是floyd的时候,不用三维数组存,最外层每循环一次直接连边就行了。
附两种图的代码
/**************************************************************
Problem:
User: BLADEVIL
Language: Pascal
Result: Accepted
Time: ms
Memory: kb
****************************************************************/ //By BLADEVIL
var
n, m, k :longint;
x, y, ss, st :longint;
l :longint;
d :array[..,..] of longint;
a, b, c :longint;
pre, other, len, cost :array[..] of longint;
last :array[..] of longint;
ans :longint;
dis, father :array[..] of longint;
flag :array[..] of boolean;
que :array[..] of longint; function min(a,b:longint):longint;
begin
if a>b then min:=b else min:=a;
end; procedure connect(a,b,c,d:longint);
begin
inc(l);
pre[l]:=last[a];
last[a]:=l;
other[l]:=b;
len[l]:=c;
cost[l]:=d;
end; function spfa:boolean;
var
h, t, q, p :longint;
cur :longint; begin
filldword(dis,sizeof(dis) div ,maxlongint div );
que[]:=ss; dis[ss]:=;
h:=; t:=;
while h<>t do
begin
h:=h mod +;
cur:=que[h];
flag[cur]:=false;
q:=last[cur];
while q<> do
begin
p:=other[q];
if len[q]> then
begin
if dis[p]>dis[cur]+cost[q] then
begin
father[p]:=q;
dis[p]:=dis[cur]+cost[q];
if not flag[p] then
begin
t:=t mod +;
que[t]:=p;
flag[p]:=true;
end;
end;
end;
q:=pre[q];
end;
end;
if dis[st]=maxlongint div then exit(false) else exit(true);
end; procedure update;
var
low :longint;
cur :longint;
begin
low:=maxlongint;
cur:=st;
while cur<>ss do
begin
low:=min(low,len[father[cur]]);
cur:=other[father[cur] xor ];
end;
cur:=st;
while cur<>ss do
begin
dec(len[father[cur]],low);
inc(len[father[cur] xor ],low);
if cost[father[cur]]<>-maxlongint div then
inc(ans,low*cost[father[cur]]);
cur:=other[father[cur] xor ];
end;
end; procedure init;
var
i, j :longint;
begin
read(n,m,k);
x:=*n+; y:=x+; ss:=y+; st:=ss+;
l:=;
connect(ss,,k,); connect(,ss,,);
for i:= to n do
begin
connect(*i,*i+,,-maxlongint div );
connect(*i+,*i,,maxlongint div );
connect(*i+,st,,); connect(st,*i+,,);
end;
filldword(d,sizeof(d) div ,maxlongint div );
for i:= to n do d[i,i]:=;
for i:= to m do
begin
read(a,b,c);
if c<d[a,b] then
begin
d[a,b]:=c; d[b,a]:=c;
end;
end;
for k:= to n do
begin
for i:= to n do
for j:= to n do
d[i,j]:=min(d[i,j],d[i,k]+d[k,j]);
connect(,*k,,d[,k]); connect(*k,,,-d[,k]);
for i:= to k- do
begin
connect(*i+,*k,,d[i,k]);
connect(*k,*i+,,-d[i,k]);
end;
end;
end; procedure main;
begin
while spfa do
update;
writeln(ans);
end; begin
init;
main;
end.
/**************************************************************
Problem:
User: BLADEVIL
Language: Pascal
Result: Accepted
Time: ms
Memory: kb
****************************************************************/ //By BLADEVIL
var
n, m, k :longint;
x, y, ss, st :longint;
l :longint;
d :array[..,..] of longint;
a, b, c :longint;
pre, other, len, cost :array[..] of longint;
last :array[..] of longint;
ans :longint;
dis, father :array[..] of longint;
flag :array[..] of boolean;
que :array[..] of longint; function min(a,b:longint):longint;
begin
if a>b then min:=b else min:=a;
end; procedure connect(a,b,c,d:longint);
begin
inc(l);
pre[l]:=last[a];
last[a]:=l;
other[l]:=b;
len[l]:=c;
cost[l]:=d;
end; function spfa:boolean;
var
h, t, q, p :longint;
cur :longint; begin
filldword(dis,sizeof(dis) div ,maxlongint div );
que[]:=ss; dis[ss]:=;
h:=; t:=;
while h<>t do
begin
h:=h mod +;
cur:=que[h];
flag[cur]:=false;
q:=last[cur];
while q<> do
begin
p:=other[q];
if len[q]> then
begin
if dis[p]>dis[cur]+cost[q] then
begin
father[p]:=q;
dis[p]:=dis[cur]+cost[q];
if not flag[p] then
begin
t:=t mod +;
que[t]:=p;
flag[p]:=true;
end;
end;
end;
q:=pre[q];
end;
end;
if dis[st]=maxlongint div then exit(false) else exit(true);
end; procedure update;
var
low :longint;
cur :longint;
begin
low:=maxlongint;
cur:=st;
while cur<>ss do
begin
low:=min(low,len[father[cur]]);
cur:=other[father[cur] xor ];
end;
cur:=st;
while cur<>ss do
begin
dec(len[father[cur]],low);
inc(len[father[cur] xor ],low);
inc(ans,low*cost[father[cur]]);
cur:=other[father[cur] xor ];
end;
end; procedure init;
var
i, j :longint;
begin
read(n,m,k);
x:=*n+; y:=x+; ss:=y+; st:=ss+;
l:=;
for i:= to n do
begin
connect(i,i+n,,); connect(i+n,i,,);
connect(i+n,y,,); connect(y,i+n,,);
connect(ss,i+n,,); connect(i+n,ss,,);
connect(i,st,,); connect(st,i,,);
end;
connect(y,x,k,); connect(x,y,,);
filldword(d,sizeof(d) div ,maxlongint div );
for i:= to n do d[i,i]:=;
for i:= to m do
begin
read(a,b,c);
if c<d[a,b] then
begin
d[a,b]:=c; d[b,a]:=c;
end;
end;
for k:= to n do
begin
for i:= to n do
for j:= to n do
d[i,j]:=min(d[i,j],d[i,k]+d[k,j]);
connect(x,k,,d[,k]); connect(k,x,,-d[,k]);
for i:= to k- do
begin
connect(i+n,k,,d[i,k]);
connect(k,i+n,,-d[i,k]);
end;
end;
end; procedure main;
begin
while spfa do
update;
writeln(ans);
end; begin
init;
main;
end.
bzoj 2324 ZJOI 营救皮卡丘 费用流的更多相关文章
- BZOJ.2324.[ZJOI2011]营救皮卡丘(费用流 Floyd)
BZOJ 洛谷 首先预处理出\(dis[i][j]\),表示从\(i\)到\(j\)的最短路.可以用\(Floyd\)处理. 注意\(i,j\)是没有大小关系限制的(\(i>j\)的\(dis[ ...
- bzoj 2324 [ZJOI2011]营救皮卡丘(floyd,费用流)
2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1777 Solved: 712[Submit][Stat ...
- BZOJ 2324: [ZJOI2011]营救皮卡丘( floyd + 费用流 )
昨晚写的题...补发一下题解... 把1~N每个点拆成xi, yi 2个. 预处理i->j经过编号不超过max(i,j)的最短路(floyd) S->0(K, 0), S->xi(1 ...
- bzoj2324 [ZJOI2011]营救皮卡丘 费用流
[ZJOI2011]营救皮卡丘 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 2653 Solved: 1101[Submit][Status][D ...
- BZOJ 2324: [ZJOI2011]营救皮卡丘(带上下限的最小费用最大流)
这道题么= =还是有些恶心的,第一次写带上下界的网络流,整个人都萌萌哒~~~ 首先先预处理得最短路后 直接用费用流做就行了。 第一次写,还是挺好写的= = CODE: #include<cstd ...
- bzoj 2324: [ZJOI2011]营救皮卡丘
#include<cstdio> #include<iostream> #include<cstring> #include<cmath> #inclu ...
- 【BZOJ 2324】[ZJOI2011]营救皮卡丘 费用流
本人实行诱骗拐卖(利用自然分层与实际意义),正解拼接补充(充分利用最大流限制(不浪费任何一个走出去的机会而不是不浪费任何一个已有的流)与问题转换) #include <cstdio> #i ...
- BZOJ-2324 营救皮卡丘 最小费用可行流+拆下界+Floyd预处理
准备一周多的期末,各种爆炸,回来后状态下滑巨快...调了一晚上+80%下午 2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec Memory Limit: 256 MB ...
- BZOJ2324: [ZJOI2011]营救皮卡丘
2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1359 Solved: 522[Submit][Stat ...
随机推荐
- 29、phonegap入门
0. PhoneGap介绍 0.1 什么是PhoneGap? PhoneGap是一个基于HTML.CSS.JS创建跨平台移动应程序的快速开发平台.与传统Web应用不同的是,它使开发者能够利用iPho ...
- 爬取妹子图(requests + BeautifulSoup)
刚刚入门爬虫,今天先对于单个图集进行爬取,过几天再进行翻页爬取. 使用requests库和BeautifulSoup库 目标网站:妹子图 今天是对于单个图集的爬取,就选择一个进行爬取,我选择的链接为: ...
- 在 C/C++ 中使用 TensorFlow 预训练好的模型—— 间接调用 Python 实现
现在的深度学习框架一般都是基于 Python 来实现,构建.训练.保存和调用模型都可以很容易地在 Python 下完成.但有时候,我们在实际应用这些模型的时候可能需要在其他编程语言下进行,本文将通过 ...
- day-11 python自带库实现2层简单神经网络算法
深度神经网络算法,是基于神经网络算法的一种拓展,其层数更深,达到多层,本文以简单神经网络为例,利用梯度下降算法进行反向更新来训练神经网络权重和偏向参数,文章最后,基于Python 库实现了一个简单神经 ...
- [转]使用 Travis CI 部署你的 Hexo 博客
之前使用Hexo的时候,都是在本地进行编译和部署的,平时使用的时候觉得没什么,只需要 hexo ghexo d 两步就可以把博客更新的内容推到GitHub Pages上.但是某些时候可能会遇到你的文件 ...
- nopcommerce商城系统--开发者常遇问题清单
原址:http://www.nopcommerce.com/docs/74/frequently-asked-development-questions.aspx 以下是开发者常见问题的清单.也介绍了 ...
- lintcode-74-第一个错误的代码版本
74-第一个错误的代码版本 代码库的版本号是从 1 到 n 的整数.某一天,有人提交了错误版本的代码,因此造成自身及之后版本的代码在单元测试中均出错.请找出第一个错误的版本号. 你可以通过 isBad ...
- 关于aspnet_regsql使用方法
aspnet_regsql命令解释 说明该向导主要用于配置SQL Server数据库,如membership,profiles等信息,如果要配置SqlCacheDependency,则需要以命令行的方 ...
- EF 4.0 升级到 6.0 问题解决办法
1.工具->库程序包管理器-> 管理解决方案的Nuget 程序包 找到EntityFramework 管理,勾选把需要进入 EF6.0的 项目,进行升级.
- springMVC笔记二
第十四章 springmvc快速入门(注解版本) 1)springmvc快速入门(传统版) 步一:创建springmvc-day02这么一个web应用 步二:导入springioc,springweb ...