最短路问题 Floyd+Dijkstra+SPFA

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了最短路问题 Floyd+Dijkstra+SPFA相关的知识,希望对你有一定的参考价值。

参考博客:https://blog.csdn.net/qq_35644234/article/details/60875818

题目来源:http://acm.hdu.edu.cn/showproblem.php?pid=2544

例题  hdu 2544

解法1.Dijkstra 

复杂度为o(n*n)   n为点的个数

从1点开始贪心地寻找最佳距离

可以求出1号点到其它点的最短距离

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn=105;
const int INF=1e9+10;
bool ins[maxn];
int ma[maxn][maxn];
int dis[maxn];
int n,m;
void dijkstra()
{

	memset(ins,0,sizeof(ins));
	for(int i=2;i<=n;i++)
		dis[i]=ma[1][i];
	dis[1]=0;
	ins[1]=1;
	while(1)
	{
		int minn=INF,index;
		for(int i=1;i<=n;i++)
		{
			if(ins[i]==0&&dis[i]<minn)
			{
				minn=dis[i];index=i;
			}
		}
		if(minn==INF)break;
		ins[index]=1;
		for(int i=1;i<=n;i++)
		{
			if(ins[i]==0&&ma[index][i]+dis[index]<dis[i])
				dis[i]=ma[index][i]+dis[index];
		}
	}
	printf("%d\n",dis[n]);

}
int main()
{

	while(scanf("%d %d",&n,&m)==2)
	{
		if(n==0&&m==0)break;
		for(int i=1;i<=n;i++)
		for(int j=1;j<=n;j++)
			ma[i][j]=INF;
		for(int i=1;i<=m;i++)
		{
			int a,b,c;
			scanf("%d %d %d",&a,&b,&c);
			ma[a][b]=min(c,ma[a][b]);
			ma[b][a]=min(c,ma[b][a]);
		}
		dijkstra();
	}
	return 0;
}

  

解法2.Floyd

复杂度o(n*n*n)

每次插入1个点,更新整个矩阵

可以求出两两之间的最短路径

#include<iostream>
#include<cstdio>
using namespace std;
const int INF=1e9+10;
const int maxn=105;
int ma[maxn][maxn];
int n,m;
void floyd()
{
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=n;j++)
			for(int k=1;k<=n;k++)
				ma[j][k]=min(ma[j][k],ma[j][i]+ma[i][k]);
	}
    printf("%d\n",ma[1][n]);
}
int main()
{
	while(scanf("%d %d",&n,&m)==2)
	{
		if(n==0&&m==0)break;
		for(int i=1;i<=n;i++)
			for(int j=1;j<=n;j++)
				ma[i][j]=INF;
		for(int i=1;i<=m;i++)
		{
			int a,b,c;
			scanf("%d %d %d",&a,&b,&c);
			ma[a][b]=min(ma[a][b],c);
			ma[b][a]=min(ma[b][a],c);
		}
		floyd();
	}
	return 0;
}

  

解法3.SPFA

复杂度o(n*v)  v为图中边的个数

贪心从1点寻找最短距离

优点是可以计算带有负边的图,缺点,复杂度高

#include<iostream>
#include<cstdio>
#include<queue>
using namespace std;
const int maxn=110;
const int INF=1e9+10;
int ma[maxn][maxn],dis[maxn];
bool ins[maxn];
int n,m;
struct Node{
	int x;
	bool	operator < (const Node &a)const
	{
		return dis[x]<dis[a.x];
	}
	Node(int a)
	{
		x=a;
	}
};
void spfa()
{
	for(int i=1;i<=n;i++)ins[i]=0;
	priority_queue<Node>que; 
	for(int i=1;i<=n;i++)
		dis[i]=INF;
	dis[1]=0;
	que.push(Node(1));
	while(que.size())
	{
		int x=que.top().x;
		que.pop();
		ins[x]=0;
		for(int i=1;i<=n;i++)
		{
			if(dis[i]>dis[x]+ma[x][i])
			{
				dis[i]=dis[x]+ma[x][i];
				if(ins[i]==0)
				{
					ins[i]=1;
					que.push(Node(i));
				}
			}
		}
	}
	printf("%d\n",dis[n]);
}
int main()
{
	while(scanf("%d %d",&n,&m)==2)
	{
		if(n==0&&m==0)break;
		for(int i=1;i<=n;i++)for(int j=1;j<=n;j++)ma[i][j]=INF;
		for(int i=1;i<=m;i++)
		{
			int a,b,c;
			scanf("%d %d %d",&a,&b,&c);
			ma[a][b]=min(ma[a][b],c);
			ma[b][a]=min(ma[b][a],c);
		}
		spfa();
	}
	return 0;
}

  

以上是关于最短路问题 Floyd+Dijkstra+SPFA的主要内容,如果未能解决你的问题,请参考以下文章

最短路问题 Floyd+Dijkstra+SPFA

最短路(Dijkstra,Floyd,Bellman_Ford,SPFA)

POJ-1847 最短路裸题,Floyd, Bellman, Dijkstra, Spfa

六度分离(floyd算法,SPFA算法,最短路—Dijkstra算法)

算法小讲堂之最短路算法(Floyd+bellman+SPFA+Dijkstra)

hdu1874 畅通project续 最短路 floyd或dijkstra或spfa