单源最短路径python
给定一个图,和一个源顶点src,找到从src到其它全部全部顶点的最短路径,图中可能含有负权值的边。算法
Dijksra的算法是一个贪婪算法,时间复杂度是O(VLogV)(使用最小堆)。可是迪杰斯特拉算法在有负权值边的图中不适用,Bellman-Ford适合这样的图。在网络路由中,该算法会被用做距离向量路由算法。Bellman-Ford也比迪杰斯特拉算法更简单。但Bellman-Ford的时间复杂度是O(VE),这要比迪杰斯特拉算法慢。(V为顶点的个数,E为边的个数)spring
算法描述网络
输入:图 和 源顶点
输出:从src到全部顶点的最短距离。若是有负权回路(不是负权值的边),则不计算该最短距离,没有意义,由于能够穿越负权回路任意次,则最终为负无穷。app
算法步骤spa
1.初始化:将除源点外的全部顶点的最短距离估计值 dist[v] ← +∞, dist[s] ←0;
2.迭代求解:反复对边集E中的每条边进行松弛操做,使得顶点集V中的每一个顶点v的最短距离估计值逐步逼近其最短距离;(运行|v|-1次)
3.检验负权回路:判断边集E中的每一条边的两个端点是否收敛。若是存在未收敛的顶点,则算法返回false,代表问题无解;不然算法返回true,而且从源点可达的顶点v的最短距离保存在 dist[v]中。code
关于该算法的证实也比较简单,采用反证法,具体参考:http://courses.csail.mit.edu/6.006/spring11/lectures/lec15.pdf
该算法是利用动态规划的思想。该算法以自底向上的方式计算最短路径。
它首先计算最多一条边时的最短路径(对于全部顶点)。而后,计算最多两条边时的最短路径。外层循环须要执行|V|-1次。blog
例子图片
一下面的有向图为例:给定源顶点是0,初始化源顶点距离全部的顶点都是是无穷大的,除了源顶点自己。由于有5个顶点,所以全部的边须要处理4次。路由
按照如下的顺序处理全部的边:(B,E), (D,B), (B,D), (A,B), (A,C), (D,C), (B,C), (E,D).
第一次迭代获得以下的结果(第一行为初始化状况,最后一行为最终结果):
当 (B,E), (D,B), (B,D) 和 (A,B) 处理完后,获得的是第二行的结果。
当 (A,C) 处理完后,获得的是第三行的结果。
当 (D,C), (B,C) 和 (E,D) 处理完后,获得第四行的结果。
第一次迭代保证给全部最短路径最多只有1条边。当全部的边被第二次处理后,获得以下的结果(最后一行为最终结果):
第二次迭代保证给全部最短路径最多只有2条边。咱们还须要2次迭代(即所谓的松弛操做),就能够获得最终结果。
为何要循环V-1次?
答:由于最短路径确定是个简单路径,不可能包含回路的,若是包含回路,且回路的权值和为正的,那么去掉这个回路,能够获得更短的路径若是回路的权值是负的,那么确定没有解了.图有n个点,又不能有回路,因此最短路径最多n-1边。又由于每次循环,至少relax一边因此最多n-1次就好了。
算法导论上的伪代码:
BELLMAN-FORD(G, w, s) 1 INITIALIZE-SINGLE-SOURCE(G, s) 2 for i ← 1 to |V[G]| - 1 3 do for each edge (u, v) ∈ E[G] 4 do RELAX(u, v, w) 5 for each edge (u, v) ∈ E[G] 6 do if d[v] > d[u] + w(u, v) 7 then return FALSE 8 return TRUE