Bellman-Ford最短路径

时间:2023-03-09 03:58:16
Bellman-Ford最短路径

对于前面说到的最短路径的求解方法,不能解决负权边的情况,而Bellman-Ford却可以

共有n个顶点,m条边,每次输入u[i],v[i],w[i],代表从u[i]到v[i]的距离是w[i],对于所有的顶点进行n-1次松弛

更多详细的内容在代码中会讲解,大家还是直接看代码吧^...^

 #include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
const int inf=;
using namespace std;
int main()
{
int u[],v[],dis[],w[],te[];
int m,n;
while(cin>>n>>m){
for(int i=;i<=m;i++){
cin>>u[i]>>v[i]>>w[i];
}
for(int i=;i<=n;i++) dis[i]=inf;///初始值设为无穷大,这样就会从第一个结点开始逐渐松弛,向后面的结点开始扩展
dis[]=;///将原点设为1
for(int i=;i<=n-;i++){
for(int k=;k<=n;k++)te[k]=dis[k];///这里把dis的初始值记录在te数组中,以便在后面我们判断是否所有的点都已经松弛完成,我们知道这个算法的最坏的情况是要松弛n-1次,但是在实际中很多情况我们提前就已经松弛完成啦,在这里加入这一步可以提前退出循环
for(int j=;j<=m;j++){
if(dis[v[j]]>dis[u[j]]+w[j])
dis[v[j]]=dis[u[j]]+w[j];
}
int check=;
for(int i=;i<=n;i++){
if(dis[i]!=te[i]){
check=;
break;
}
}
if(check==)break;///当在新一轮的松弛中我们发现所有点的dis值都没有改变,这就表明所有的点都已经松弛完成啦
}
int flag=;
for(int i=;i<=m;i++)
if(dis[v[i]]>dis[u[i]]+w[i])flag=;///当松弛完毕后如果还能找到可以继续松弛的边的话就代表含有负权边,我们通常用这种方法检查是否含有负权回路
if(flag==)cout<<"此图含有负权回路";
else { for(int i=;i<=n;i++)cout<<dis[i]<<" ";
cout<<endl;
}
}
}