xiaoguozi's Blog
Pay it forword - 我并不觉的自豪,我所尝试的事情都失败了······习惯原本生活的人不容易改变,就算现状很糟,他们也很难改变,在过程中,他们还是放弃了······他们一放弃,大家就都是输家······让爱传出去,很困难,也无法预料,人们需要更细心的观察别人,要随时注意才能保护别人,因为他们未必知道自己要什么·····
最小费用最大流 修改的dijkstra + Ford-Fulksonff算法
修改的dijkstra其实和Johnson算法的思想是一致的。
 
一个求最小费用最大流的朴素算法是这样的:
1 求最小费用增广路
2 判断是否存在增广路,否的话算法终止。
3 增加增广路上边的流量
4 在增广路上添加必要的逆向负权边
5 goto 1
 
因为负权边的存在,求最小费用增广路就不可以用dijkstra算法。当然,我们可以用bellman
-ford算法,可是这样的话求一次最短路的时间代价就是O(e*n),e是边数,n是顶点数。代价大了点,如果能用dijkstra算法就好了。利用Johnson算法的思想,这是可以做到的。
 
第一次求最短路可以用dijkstra算法(如果一开始就有负权边,那就用bellman
-ford算法,这没关系),求出源点到所有点的距离,嗯,我说的距离是指路径上边的费用之和的最小值。注意,要求出到所有点的距离,而不是求出到汇点的距离就完事了。
假设有一条边u
->v,源点到u的距离是d[u],到v的距离是d[v],边的费用(权值)是w(u,v)。很显然,d[u]+w(u,v)>=d[v],不然的话,你会发现一条更好的路径从源点到v。问题是,什么时候取等呢?当u->v在v的最优路径上,范围说小一点,当u->v在从源点到汇点的最优路径,即最小费用增广路上。
好的,如果u
->v被你增载了,你要开始添负权边v->u了,权值取负,就是-w(u,v)。负权就是讨厌,是正的就好了,dijkstra算法就可以再用了。怎么办呢,把负权边加个权值,让它非负。要加多少呢,d[v]-d[u]。当然不能只加一条边,对所有边,无论原有的还是新添的,按这个规则加,构造一个新的图:
                对边a
->b,新的边权w'(a,b)=w(a,b)+d[a]-d[b]
现在来看看你的杰作:
                对原来的边u
->v, w'(u,v)=w(u,v)+d[u]-d[v]: 记得么d[u]+w(u,v)>=d[v], 所以 w'(u,v) >= 0
                对新加的负权边v
->u, w'(v,u)=w(v,u)+d[v]-d[u]=-w(u,v)+d[v]-d[u]: 记得么d[u]+w(u,v)==d[v],这里可是取等号的,所以w'(v,u) == 0
哈哈,这下所有边又是非负的了。
可是,问题是,为啥不每个边加个足够大的正数,这样不是所有边也都是正的了么。仔细想想,边权为啥要为正,不就是为了求源点到汇点的最短路方便么,可是,都加大正数的话,你求出的最短路和原来图的最短路能一致么,不能,为啥,画个三角形,自己想想。可是,我的方法就能一致么,能。我证明给你看。
 
假设从源点s到汇点t有一条路径s
->a->b->c->d.->t,在原图中的路径长度为
                 w(s,a)
+w(a,b)+w(b,c)++w(x,t)
在新图中的路径为
                 w
'(s,a)+w'(a,b)+w'(b,c)+w'(x,t)
展开来就是
                 w(s,a)
+d[a]-d[s]+w(a,b)+d[b]-d[a]+w(c,d)+d[d]-d[b]+.+w(x,t)+d[t]-d[x]
消阿消,d[a]和
-d[a],d[b]和-d[b]d[x]和-d[x],剩下什么呢:
                 w(s,a)
+w(a,b)+w(b,c)++w(x,t)+d[t]-d[s]
噢,不就比原图中多d[t]
-d[s]么(其实d[s]==0)。这可是对所有s到t的路径都成立的,既然所有路径,在新图中的权值都比在原图中的权值多了d[t],那么,新图的最短路,也就对应原图的最短路,只不过路径长度多了d[t],这不仅对t成立,对所有节点u都成立,只不过新图中到u的最短路长度比原图多了d[u]。
 
好,用dijkstra算法,第二次求出最短路。然后求出新的d’[u],然后添加新的边,然后准备第三次的dijkstra算法。。。为什么第二次可以这样做,第三次还可以这样做,第三次的原图可能有很多负权边啊?我可没说过w(u,v)
>=0这样的限制,所以,即使原图有负权边还是可以这样做的。
 
好了,第一次dijkstra算法(或者bellman
-ford算法,如果有负权边的话,只用一次,不会成为瓶颈的),然后每次求最小增广路用一次修改的dijkstra算法。这个算法求最小费用最大流复杂度是O(m*n*n), m是最大流量,或者是求增广路次数的上界。最后,如果用这个算法来求最优匹配问题,复杂度是O(n^3)的。
posted on 2008-08-03 20:49 小果子 阅读(5060) 评论(9)  编辑 收藏 引用

FeedBack:
# re: 最小费用最大流
2008-08-30 12:24 | newmyl
最小费用最大流,我用Dijkstra过poj2516最后是900多ms,但我用Spfa过这题,我只用了700多ms,  回复  更多评论
  
# re: 最小费用最大流
2009-09-27 20:19 | SwordHoly
大牛能给个代码吗,我邮箱jiangmingh@126.com,万分感谢,自己写的实在是头大了,因为d值一直要变,不知道结果怎么算了  回复  更多评论
  
# re: 最小费用最大流
2009-11-12 09:51 | janlim
你好,能给个程序吗?我的邮箱是linjunemail@163.com,和输入的格式。。先谢谢了、、  回复  更多评论
  
# re: 最小费用最大流
2010-03-08 10:06 | nuanbaby
你好,能同样发我个程序和输入的格式吗?我的邮箱是wszbd_lz@126.com~~~

谢谢啦~~~  回复  更多评论
  
# re: 最小费用最大流[未登录]
2010-12-07 20:54 | along
大哥讲得很具体嘛,赞一个  回复  更多评论
  
# re: 最小费用最大流
2012-03-08 18:15 | 旷志鑫
第一次就有负边,好像一定要用bellman-ford求出距离,  回复  更多评论
  
# re: 最小费用最大流
2012-03-08 18:16 | 旷志鑫
可以把原先的先保存起来,以后就不用变了  回复  更多评论
  
# re: 最小费用最大流
2012-04-04 11:51 | xw
谢谢!  回复  更多评论
  
# re: 最小费用最大流
2015-02-20 21:42 | Spfa
能不能发个代码呢>>最好是Fibonacci堆优化的Dijkstra
C++ 的
谢谢!  回复  更多评论
  

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理