原文地址:http://www.wutianqi.com/?p=1912
相关文章:
1.Dijkstra算法:
http://www.wutianqi.com/?p=1890
2.Floyd算法:
http://www.wutianqi.com/?p=1903
Dijkstra算法是处理单源最短路径的有效算法,但它局限于边的权值非负的情况,若图中出现权值为负的边,Dijkstra算法就会失效,求出的最短路径就可能是错的。这时候,就需要使用其他的算法来求解最短路径,Bellman-Ford算法就是其中最常用的一个。该算法由美国数学家理查德•贝尔曼(Richard Bellman, 动态规划的提出者)和小莱斯特•福特(Lester Ford)发明。Bellman-Ford算法的流程如下:
给定图G(V, E)(其中V、E分别为图G的顶点集与边集),源点s,
可知,Bellman-Ford算法寻找单源最短路径的时间复杂度为O(V*E).
首先介绍一下松弛计算。如下图:
松弛计算之前,点B的值是8,但是点A的值加上边上的权重2,得到5,比点B的值(8)小,所以,点B的值减小为5。这个过程的意义是,找到了一条通向B点更短的路线,且该路线是先经过点A,然后通过权重为2的边,到达点B。
当然,如果出现一下情况
则不会修改点B的值,因为3+4>6。
Bellman-Ford算法可以大致分为三个部分
第一,初始化所有点。每一个点保存一个值,表示从原点到达这个点的距离,将原点的值设为0,其它的点的值设为无穷大(表示不可达)。
第二,进行循环,循环下标为从1到n-1(n等于图中点的个数)。在循环内部,遍历所有的边,进行松弛计算。
第三,遍历途中所有的边(edge(u,v)),判断是否存在这样情况:
d(v) > d (u) + w(u,v)
则返回false,表示途中存在从源点可达的权为负的回路。
之所以需要第三部分的原因,是因为,如果存在从源点可达的权为负的回路。则 应为无法收敛而导致不能求出最短路径。
考虑如下的图:
经过第一次遍历后,点B的值变为5,点C的值变为8,这时,注意权重为-10的边,这条边的存在,导致点A的值变为-2。(8+ -10=-2)
第二次遍历后,点B的值变为3,点C变为6,点A变为-4。正是因为有一条负边在回路中,导致每次遍历后,各个点的值不断变小。
在回过来看一下bellman-ford算法的第三部分,遍历所有边,检查是否存在d(v) > d (u) + w(u,v)。因为第二部分循环的次数是定长的,所以如果存在无法收敛的情况,则肯定能够在第三部分中检查出来。比如
此时,点A的值为-2,点B的值为5,边AB的权重为5,5 > -2 + 5. 检查出来这条边没有收敛。
所以,Bellman-Ford算法可以解决图中有权为负数的边的单源最短路径问。
个人感觉算法导论讲解很不错,把这一章贴出来和大家分享:
24.1 The Bellman-Ford algorithm
The Bellman-Ford algorithm solves the single-source shortest-paths problem in the general case in which edge weights may be negative. Given a weighted, directed graph G = (V, E) with source s and weight function w : E → R, the Bellman-Ford algorithm returns a boolean value indicating whether or not there is a negative-weight cycle that is reachable from the source. If there is such a cycle, the algorithm indicates that no solution exists. If there is no such cycle, the algorithm produces the shortest paths and their weights.
The algorithm uses relaxation, progressively decreasing an estimate d[v] on the weight of a shortest path from the source s to each vertex v ∈ V until it achieves the actual shortest-path weight δ(s, v). The algorithm returns TRUE if and only if the graph contains no negative-weight cycles that are reachable from the source.
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
Figure 24.4 shows the execution of the Bellman-Ford algorithm on a graph with 5 vertices. After initializing the d and π values of all vertices in line 1, the algorithm makes |V| – 1 passes over the edges of the graph. Each pass is one iteration of the for loop of lines 2-4 and consists of relaxing each edge of the graph once. Figures 24.4(b)-(e) show the state of the algorithm after each of the four passes over the edges. After making |V|- 1 passes, lines 5-8 check for a negative-weight cycle and return the appropriate boolean value. (We’ll see a little later why this check works.)
Figure 24.4: The execution of the Bellman-Ford algorithm. The source is vertex s. The d values are shown within the vertices, and shaded edges indicate predecessor values: if edge (u, v) is shaded, then π[v] = u. In this particular example, each pass relaxes the edges in the order (t, x), (t, y), (t, z), (x, t), (y, x), (y, z), (z, x), (z, s), (s, t), (s, y). (a) The situation just before the first pass over the edges. (b)-(e) The situation after each successive pass over the edges. The d and π values in part (e) are the final values. The Bellman-Ford algorithm returns TRUE in this example.
The Bellman-Ford algorithm runs in time O(V E), since the initialization in line 1 takes Θ(V) time, each of the |V| – 1 passes over the edges in lines 2-4 takes Θ(E) time, and the for loop of lines 5-7 takes O(E) time.
以下是Bellman-Ford代码:
#include <iostream> #include <stdlib.h> #include <string.h> using namespace std; /* Bellman-Ford:计算带有负权值的最短路径,注意:带有负环的最短路径无法求得 算法思想: S1:初始化除源点外所有点的最短距离为无穷,源点距离为0 S2:执行|V-1|次循环,每次对所有边进行松弛操作,即:如果d[u] + w(u,v) < d[v],则令d[v] = d[u] + w(u,v) S3:遍历每条边,对每条边检测是否可以继续松弛,即:如果d[u] + w(u,v) < d[v],说明存在负环,无法计算最短距离;若每条边都不可以松弛, 则d[i]表明源点到节点i的最短距离 输出: 如果存在负环,输出:存在负环 如果不存在负环,按照节点下标从小到大的顺序,输出:源点到各个节点的最短距离 距离数组的大小与顶点个数相同 输入: 3(边数) 3(顶点数) 0(源点) 0 1 5 1 2 3 2 0 -10 输出: 存在负环 输入: 10 5 0 0 1 6 0 2 7 1 2 8 1 3 5 1 4 -4 2 3 -3 2 4 9 3 1 -2 4 0 2 4 3 7 输出: 2 7 4 -2 节点1到源点0的最短距离:2,路径:0 2 3 1 节点2到源点0的最短距离:7,路径:0 2 节点3到源点0的最短距离:4,路径:0 2 3 节点4到源点0的最短距离:-2,路径:0 2 3 4 */ const int N = 10000; int d[N]; int pre[N]; typedef struct Edge { Edge(){} Edge(int iBeg , int iEnd , int iWeight):_iBeg(iBeg),_iEnd(iEnd),_iWeight(iWeight){} void setEdge(int iBeg , int iEnd , int iWeight) { _iBeg = iBeg; _iEnd = iEnd; _iWeight = iWeight; } int _iBeg, _iEnd; int _iWeight; }Edge; bool bellmanFord(int iStart ,int iVertexNum , int iEdgeNum , Edge* pEdge) { if(!pEdge) { return false; } //设定起始节点的前向节点为本身,用于打印单源最短路径 memset(pre, -1 , sizeof(pre)); pre[iStart] = iStart; //初始化,对除源点外的顶点距离设置为无穷大 for(int i = 0 ; i < iVertexNum ; i++ ) { if(i != iStart) { d[i] = INT_MAX; } else { d[i] = 0; } } //迭代|V|-1次,每次对所有边进行松弛 for(int i = 0 ; i < iVertexNum - 1 ; i++) { //对每条边遍历,进行松弛处理 for(int j = 0 ; j < iEdgeNum ; j++) { int u = pEdge[j]._iBeg; int v = pEdge[j]._iEnd; int iWeight = pEdge[j]._iWeight; //如果起点的距离值+边的权重<终点的距离值,就进行松弛处理 if(d[u] + iWeight < d[v]) { d[v] = d[u] + iWeight; //记录每次松弛后终点的前面的端点为 pre[v] = u; } } } //负权检测:遍历所有边,检测是否还可以继续松弛,如果是,表明存在负环,无法计算单源最短路径 for(int j = 0 ; j < iEdgeNum ; j++) { int u = pEdge[j]._iBeg; int v = pEdge[j]._iEnd; int iWeight = pEdge[j]._iWeight; //如果起点的距离值+边的权重<终点的距离值,就进行松弛处理 if(d[u] + iWeight < d[v]) { return false; } } return true; } //打印路径,参数root:当前端点下标,采用先递归后打印的方式 void printPath(int root) { //递归基 if(pre[root] == root) { //打印起始节点 cout << root << " "; return; } printPath(pre[root]); cout << root << " " ; } void process() { int iVertexNum , iEdgeNum , iStart; Edge edgeArr[N]; while(cin >> iEdgeNum >> iVertexNum >> iStart) { //生成边 for(int i = 0 ; i < iEdgeNum ; i++) { cin >> edgeArr[i]._iBeg >> edgeArr[i]._iEnd >> edgeArr[i]._iWeight; } //调用贝尔曼-福特算法 if(bellmanFord(iStart ,iVertexNum , iEdgeNum , edgeArr)) { for(int i = 0 ; i < iVertexNum ; i++ ) { if(i != iStart) { cout << "节点" << i << "到源点" << iStart << "的最短距离:" << d[i] << ",路径:"; printPath(i); cout << endl; } } } else { cout << "存在负环" << endl; } } } int main(int argc,char* argv[]) { process(); system("pause"); return 0; }
补充:
考虑:为什么要循环V-1次?
答:因为最短路径肯定是个简单路径,不可能包含回路的,
如果包含回路,且回路的权值和为正的,那么去掉这个回路,可以得到更短的路径
如果回路的权值是负的,那么肯定没有解了
图有n个点,又不能有回路
所以最短路径最多n-1边
又因为每次循环,至少relax一边
所以最多n-1次就行了