Bellman-ford 算法适用于含有负权边的最短路求解,复杂度是O( VE ),其原理是依次对每条边进行松弛操作,重复这个操作E-1次后则一定得到最短路,如果还能继续松弛,则有负环。这是因为最长的没有环路的路,也只不过是V个点E-1条边构成的,所以松弛E-1次一定能得到最短路。因此这个算法相比 Dijkstra 首先其是对边进行增广,其次它能检测出负环的存在(若负环存在,那么最短路是取不到的,因为可以一直绕着这个负环将最小路径值不断缩小),这个弥补了 Dijkstra 的不足,但是其算法跑的比较慢,因此为了追求速度往往采用其“队列优化版”==>SPFA,因此想要理解SPFA最好先看看Bellman-ford算法。
SPFA 算法适用于含有负权边的最短路求解,其复杂度并没有网上传的那么神乎在理想情况下有论文指出其复杂度为O(kE)且k是一个约小于2的常数,但是在一些稠密图下其算法性能还是会退化到和 Bellman-ford 一样的 O( VE ),所以在稠密图下建议使用 Dij + Heap 优化的版本,稀疏图下 SPFA 还是很给力的!在 Bellman-ford 中发现啊最外层的 N-1 次循环未免盲目、实际上被松弛过的点我们希望其去继续松弛其他点,这样我们用队列将被松弛过的点存起来以便下一次继续松弛其他点,具体原理和做法可以参考下面的链接,顺便一提,SPFA还有两个优化==> SLF 与 LLL,具体也不阐述了。本文主要给出模板!
算法原理 or 学习参考链接 : 点我 、点我啦 、 点嘛!
Bellman-ford模板
///POJ 2387为例 #include<bits/stdc++.h> using namespace std; const int maxn = 1e3 + 10; const int INF = 0x3f3f3f3f; struct EdgeNode{ int from, to, w; }; EdgeNode Edge[maxn*maxn]; int Dis[maxn]; int N, M, cnt; inline void init() { for(int i=0; i<=N; i++) Dis[i] = INF; cnt = 0; } bool BellmanFord(int st) { Dis[st] = 0; for(int i=0; i<N; i++){///N-1 次循环后肯定能找出最短路 bool Changed = false; int to, from, weight; for(int j=0; j<cnt; j++){ to = Edge[j].to, from = Edge[j].from, weight = Edge[j].w; if(Dis[from]!=INF && Dis[to] > Dis[from] + weight){ Changed = true; Dis[to] = Dis[from] + weight; ///pre[to] = j; //Record paths } } if(!Changed) return true;///如果没有边可以继续松弛了,说明算法结束且无负环 if(i==N && Changed) return false;///有负环 } return false; ///一般来说绝无可能执行到这一步 } int main(void) { while(~scanf("%d %d", &M, &N)){ init(); int from, to, weight; for(int i=0; i<M; i++){ scanf("%d %d %d", &from, &to, &weight); Edge[cnt].from = from; Edge[cnt].to = to; Edge[cnt].w = weight; cnt++; Edge[cnt].to = from; Edge[cnt].from = to; Edge[cnt].w = weight; cnt++; } BellmanFord(1); printf("%d ", Dis[N]); } return 0; }
SPFA模板( SLF 优化版 )
///POJ 2387为例 #include <iostream> #include <cstdio> #include <cmath> #include <queue> #include <string.h> using namespace std; const int INF=0x3f3f3f3f; const int maxn = 1e3 + 10; struct EdgeNode{ int v, w, nxt; }; EdgeNode Edge[maxn*maxn]; bool vis[maxn]; int Head[maxn], Dis[maxn], cnt; int N, M; /// int PushCnt[maxn]; ///记录每一个节点的入队次数、方便判断负环 inline void init() { for(int i=0; i<=N; i++) ///PushCnt[i] = 0; Head[i] = -1, Dis[i] = INF, vis[i] = false; cnt = 0; } inline void AddEdge(int from, int to, int weight) { Edge[cnt].w = weight; Edge[cnt].v = to; Edge[cnt].nxt = Head[from]; Head[from] = cnt++; } void SPFA(int st)///若要判断负环、改为 bool { deque<int> que; que.push_back(st); vis[st]=true; Dis[st]=0; while (!que.empty()) { int T=que.front(); que.pop_front(); vis[T]=false; for (int i=Head[T]; i!=-1; i=Edge[i].nxt) { int v=Edge[i].v; int w=Edge[i].w; if (Dis[v]>Dis[T]+w){ Dis[v]=Dis[T]+w; ///p[v] = T; if (!vis[v]){ ///if(++PushCnt[v] > N) return false; //有负环 vis[v]=true; if(!que.empty() && Dis[v] < Dis[que.front()]) que.push_front(v); else que.push_back(v); //que.push_back(v); ///无SLF优化是这样写的 } } } } /// return true; } int main(void) { while(~scanf("%d %d", &M, &N)){ init(); int from, to, weight; for(int i=0; i<M; i++){ scanf("%d %d %d", &from, &to, &weight); AddEdge(from, to, weight); AddEdge(to, from, weight); } SPFA(1); printf("%d ", Dis[N]); } return 0; }