最短路(Dijkstra, Bellman-Ford, SPFA, Floyd)

2024-04-28 15:52

本文主要是介绍最短路(Dijkstra, Bellman-Ford, SPFA, Floyd),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最短路

Dijkstra算法(复杂度 O ( m l o g n ) O(mlog n) O(mlogn)/ O ( n m l o g n ) O(nmlogn) O(nmlogn))–不能有负权边,不能有负权环,单源最短路径( O ( m l o g n ) O(mlog n) O(mlogn)),多源最短路径( O ( n m l o g n ) O(nmlogn) O(nmlogn))。
Bellman-Ford算法(复杂度 O ( n m ) O(nm) O(nm))—能有负权边,不能有负权环,单源最短路径,有边数限制的最短路
SPFA算法(复杂度 O ( k m ) O(km) O(km))—能有负权边,不能有负权环,单源最短路径
Floyd(复杂度 O ( n 3 ) O(n^3) O(n3))—可以求多源最短路,可以处理负权图,不可处理有负权环的图!

Dijkstra

Dijkstra-数据结构:三个数据结构

int dis[10004],vis[10004];
priority_queue<pair<int,int>,vector<pair<int,int>>,greater<pair<int,int>>> q;

//各个数据结构的作用讲解:
dis[]:
vis[]:模拟了一个集合,vis[i]=1表示顶点 i 被加进vis集合中,vis[i]=0表示顶点 i 未被加进集合中。
q:注意优化队列一定要放pair<int,int>,并且前面一定要是距离,后面才是点!被放进去的点是有可能可以用来更新其他点的。

Dijkstra-初始化:

memset(dis,0x3f,sizeof dis); //dis数组应该初始化为无穷 
memset(vis,0,sizeof vis);

Dijkstra-预处理:

dis[i]=0; 
//vis[i]=1; //注意!不能加这一步。 
q.push({0,i}); //<长度,点>

Dijkstra-核心代码:

while(q.size())
{pair<int,int> tmp=q.top();q.pop();int t=tmp.second;if(vis[t]==1) continue;vis[t]=1;for(int i=head[t];i;i=nxt[i]){if(vis[to[i]]==0 && dis[t]+w[i]<dis[to[i]]){dis[to[i]]=dis[t]+w[i];q.push({dis[to[i]],to[i]});}}
}

Dijkstra-最后得到的东西:
一个dis[]数组,记录着起点i到其余各顶点的最短距离。

算法复杂度:
应该是O(nlog n)

好题:
旅行

旅行从城市 1 1 1到城市 n n n, 一天游玩一个城市。点与点之间有一些固定费用的路,但每隔一天需要多付出 w w w的费用测核酸。求最少花费。

#include <bits/stdc++.h>
using namespace std;#define int long long
#define pii pair<int,int>
#define se second
#define fi first
#define pb push_back
const int N=1e5+5;
int n,m,x;
int dis[N][2],vis[N][2];
vector<pii> v[N]; //<点,距离> 
priority_queue<pair<int,pii>,vector<pair<int,pii>>,greater<pair<int,pii>>> q;void Dij()
{memset(dis,0x3f,sizeof dis);memset(vis,0,sizeof vis);dis[1][0]=0;q.push({0,{1,0}}); //<距离,点,标志>while(q.size()){auto tmp=q.top();q.pop();int t=tmp.se.fi;int f=tmp.se.se;if(vis[t][f]==1) continue;vis[t][f]=1;for(int i=0;i<v[t].size();++i) //遍历t点相连的点{if(dis[t][f]+v[t][i].se+(f?0:x)<dis[v[t][i].fi][f^1]){dis[v[t][i].fi][f^1]=dis[t][f]+v[t][i].se+(f?0:x);q.push({dis[v[t][i].fi][f^1],{v[t][i].fi,f^1}});}} }
}signed main(){scanf("%lld%lld%lld",&n,&m,&x); int tt,ttt,w;for(int i=1;i<=m;++i){scanf("%lld%lld%lld",&tt,&ttt,&w); v[tt].pb({ttt,w});v[ttt].pb({tt,w});}Dij();//len[1]=1 printf("%lld\n",min(dis[n][1],dis[n][0]));
}

心得:考虑清楚一个点可以由什么点去松弛得到。

Floyd

参考 算法思想:从第1个到第n个点依次加入松弛计算,每个点加入进行试探枚举是否有路径长度被更改(自己能否更新路径)。顺序加入(k枚举)松弛的点时候,需要遍历图中每一个点对(i,j双重循环),判断每一个点对距离是否因为加入的点而发生最小距离变化,如果发生改变(变小),那么两点(i,j)距离就更改。

应用一:求任意两点间的最短路
应用二:删边,从而降低复杂度 2021CCPC女生赛 C. 连锁商店

//Floyd(弗洛伊得)算法 
特别注意!!!两点之间的距离:
D[i][j]=0  (若i==j) 
D[i][j]=inf  (若i!=j且无边相连)
D[i][j]=W[i][j]  (若i!=j且有边相连)算法描述:
for(k=1;k<=N;k++) //中间点for(i=1;i<=N;i++) //起点 for(j=1;j<=N;j++) //终点 if(D[i][k]+D[k][j]<D[i][j])D[i][j]=D[i][k]+D[k][j]; //更新最小值结束:
D[i][j]就是从i到j的最短路径的长度。 优化:
第一层循环:for(i=1;i<=N/2;i++) ——然后 D[i][j] = D[j][i] = ...如果是固定源点的话,就 去掉第一层循环。 
Bellman-Ford

有边数限制的最短路

题意:有n个点m条边的图(图中可能存在重边和自环, 边权可能为负数,可能存在负权回路),求从1号点到n号点的最多经过k条边的最短距离。如果不存在满足条件的路径,则输出 impossible

参考代码:

#include <bits/stdc++.h>
using namespace std;#define int long long
const int inf=0x3f3f3f3f;
const int N=510,M=1e4+4;
int n,m,k,dis[N],tmp[N];struct edge{int x,y,z;
}e[M];void bellman_ford(){ //答案存在dis里  for(int i=1;i<=n;++i) dis[i]=inf;dis[1]=0;for(int i=0;i<k;++i){ //限制k条边,松弛k次for(int j=1;j<=n;++j) tmp[j]=dis[j];for(int j=1;j<=m;++j){ //遍历m条边int u=e[j].x,v=e[j].y,w=e[j].z;dis[v]=min(dis[v],tmp[u]+w);}}
} signed main(){ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);cin>>n>>m>>k;for(int i=1;i<=m;++i) cin>>e[i].x>>e[i].y>>e[i].z;bellman_ford(); if(dis[n]>(0x3f3f3f3f/2)) cout<<"impossible\n"; //有可能边是负数 else cout<<dis[n]<<'\n';
}
SPFA

算法步骤

数据结构:
Dis 代表S到I点的当前最短距离。
Fa 代表S到I的当前最短路径中I之前的一个点的编号。
维护一个队列,里面存放所有需要进行迭代的点。初始时队列中只有一个点S。
用一个布尔数组记录每一个点是否在队列中。

初始化:
开始时,Dist全部为inf,只有Dist[S]=0,Fa全部为0 。

过程:

每次迭代,取出队头的点v,依次枚举从v出发的边v->u,设边的长度为len,判断Dist[v]+len是否小于Dis[u],若小于则改进Dist[u],将Fa[u]记为v,并且由于S到u的最短距离变小了,有可能u可以改进其它的点,所以若u不在队列中,就将它放入队尾。这样一直迭代下去,直到队列变空,也就是说S到所有结点的距离都确定下来,结束算法。

算法复杂度

在平均情况下,SPFA算法的期望时间复杂度O(KM) 。M为边数,K是每个点平均入队次数。

但是,有很特殊的情况,这张图是一个棋盘格子,网格图,你的SPFA是很有可能被卡到NM的然后它就死了。所以在OI里面,有一句话,关于SPFA,它死了。

所以没有负权边,就尽量不要用它了,万一被卡了。

对比Bellman-Ford:
Bellman-Ford:每一次松弛的时候Bellman-Ford都要枚举所有的点,而其实很多点都是不需要被枚举的,所以会有很多的无效枚举,使得算法效率降低。
SPFA:其实每次松弛的时候只需要枚举与上次被松弛的点相连的点就可以了。

求最短路(含负权边)
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int N = 510,M = 10010;int n,m,k;
int dist[N],backup[N]; //backup数组为上次
struct edges
{int a,b,w; // a->b权值为w的边 
}edge[M];int bellman_ford()
{memset(dist,0x3f,sizeof(dist));dist[1] = 0;for(int i=0; i<k; i++) {memcpy(backup,dist,sizeof(dist)); //备份for(int j=0; j<m; j++)   // 枚举所有边 {int a = edge[j].a, b = edge[j].b, w=edge[j].w;	dist[b] = min(dist[b],backup[a]+w); // 用备份更新 }}if(dist[n] > 0x3f3f3f3f/2) return -1;return dist[n];
}
int main()
{cin >> n >> m >> k;for(int i=0; i<m; i++){int a,b,w;cin >> a >> b >> w;edge[i] = {a,b,w}; 	}	int t = bellman_ford();if(t == -1) cout << "impossible";else cout << t;return 0;
} 
判断是否存在负权环

若一个点入队次数超过n,则有负权环。因为它最多被n-1个点(其他所有点)更新。所以要看是否存在负权环,用一个cnt[]数组,记录每个点的入队次数。

裸题:洛谷-P3385 【模板】负环

#include <bits/stdc++.h>
using namespace std;const int N = 2e3+10;
const int M = 3e4+10; //开这个数,要细心一点,像无向图,它是对称存储的,如m=2,其实是需要存4条边,当m=2e3时,其实会有4e3条边,如果只开2e3会出现各种错,像re、wa等 
const int inf = 0x3f3f3f3f;
int num;
int head[N];
int to[M];
int w[M];
int ne[M];int T,n,m,dis[N],vis[N],cnt[N];inline void addedge(int x,int y,int z){to[++num]=y;w[num]=z;ne[num]=head[x];head[x]=num;
}void SPFA(){queue<int> q;//初始化memset(dis,0x3f,sizeof dis);dis[1]=0;vis[1]=1;q.push(1);++cnt[1];while(q.size()){int u=q.front();q.pop();vis[u]=0;for(int i=head[u];i;i=ne[i]){if(dis[u]+w[i]<dis[to[i]]){dis[to[i]]=dis[u]+w[i];if(vis[to[i]]==0){q.push(to[i]);vis[to[i]]=1;++cnt[to[i]]; //入队一次就加一次。 if(cnt[to[i]]>=n){cout << "YES" << endl;return;}}}}}cout << "NO" << endl;
} int main(){cin >> T;while(T--){memset(head,0,sizeof head);memset(to,0,sizeof to);memset(ne,0,sizeof ne);memset(w,0,sizeof w);num=0;memset(vis,0,sizeof vis);memset(cnt,0,sizeof cnt);cin >> n >> m;for(int i=1;i<=m;i++){int x,y,z;cin >> x >> y >> z;addedge(x,y,z);if(z>=0)addedge(y,x,z);}SPFA();	}return 0;
}

这篇关于最短路(Dijkstra, Bellman-Ford, SPFA, Floyd)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/943654

相关文章

poj 1511 Invitation Cards(spfa最短路)

题意是给你点与点之间的距离,求来回到点1的最短路中的边权和。 因为边很大,不能用原来的dijkstra什么的,所以用spfa来做。并且注意要用long long int 来存储。 稍微改了一下学长的模板。 stack stl 实现代码: #include<stdio.h>#include<stack>using namespace std;const int M

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int

poj 1502 MPI Maelstrom(单源最短路dijkstra)

题目真是长得头疼,好多生词,给跪。 没啥好说的,英语大水逼。 借助字典尝试翻译了一下,水逼直译求不喷 Description: BIT他们的超级计算机最近交货了。(定语秀了一堆词汇那就省略吧再见) Valentine McKee的研究顾问Jack Swigert,要她来测试一下这个系统。 Valentine告诉Swigert:“因为阿波罗是一个分布式共享内存的机器,所以它的内存访问

uva 10801(乘电梯dijkstra)

题意: 给几个电梯,电梯0 ~ n-1分别可以到达很多层楼。 换乘电梯需要60s时间。 问从0层到target层最小的时间。 解析: 将进入第0层的电梯60s也算上,最后减。 坑点是如果target为0输出0。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algori

uva 10099(floyd变式)

题意: 有一个导游要带着一群旅客从一个城市到达另一个城市,每个城市之间有最大的旅客流量限制。 问最少几趟能将这些旅客从一个城市搞到另一个城市。 解析: 用floyd找出最小流量中的最大边,然后次数就是   ceil(总人数 / 最大承载量 - 1),-1的意思是导游每次也要在车上。 ps.老司机哭晕在厕所 代码: #include <iostream>#includ

uva 10048(floyd变式)

题意: 求两个点之间经过的路径中最大噪声最小的值。 解析: floyd的变式,每次取g[i][k] g[k][j]中的大边与当前边g[i][j]比较,取小。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#includ

poj 3159 (spfa差分约束最短路) poj 1201

poj 3159: 题意: 每次给出b比a多不多于c个糖果,求n最多比1多多少个糖果。 解析: 差分约束。 这个博客讲差分约束讲的比较好: http://www.cnblogs.com/void/archive/2011/08/26/2153928.html 套个spfa。 代码: #include <iostream>#include <cstdio>#i

poj 1734 (floyd求最小环并打印路径)

题意: 求图中的一个最小环,并打印路径。 解析: ans 保存最小环长度。 一直wa,最后终于找到原因,inf开太大爆掉了。。。 虽然0x3f3f3f3f用memset好用,但是还是有局限性。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#incl

hdu 3790 (单源最短路dijkstra)

题意: 每条边都有长度d 和花费p,给你起点s 终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。 解析: 考察对dijkstra的理解。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstrin

poj 3169 spfa 差分约束

题意: 给n只牛,这些牛有些关系。 ml个关系:fr 与 to 牛间的距离要小于等于 cost。 md个关系:fr 与 to 牛间的距离要大于等于 cost。 隐含关系: d[ i ] <= d[ i + 1 ] 解析: 用以上关系建图,求1-n间最短路即可。 新学了一种建图的方法。。。。。。 代码: #include <iostream>#include