最短路算法总结(*【模板】)

1.Dijkstra算法:(计算正权图上的单源最短路  single-source shortest paths (sssp) )从单个节点出发到所有节点的最短路。复杂度:O(n*n)

该算法适用于:有向图和无向图。

1). O(n^2)的实现:邻接矩阵map存储实现,INF表示无穷大

void Dijkstra(int s, int e, int n) //从s开始到e点的最短路,有n个节点,编号:0-->n-1.

{

    memset(vis, 0, sizeof(vis));

    int i, j;

    for(i=0; i<n; i++)

        dis[i]=(i==0?0:INF);

    for(i=0; i<n; i++)

    {

        int mm=INF;  int pos;

        for(j=0; j<n; j++)

        {

            if(!vis[j] && dis[j]<m)

                m=dis[pos=j];

        }

        vis[pos]=1;

        for(j=0; j<n; j++)

        {

            d[j]=min(dis[j], dis[pos]+map[pos][j] );

        }

    }

    printf("%d\n", dis[e]); //假设最短路一定存在

}

 优化时间复杂度的dijkstra算法:

 代码:

#include <stdio.h>

#include <string.h>

#include <stdlib.h>

#include <algorithm>

#include <vector>

#include <queue>

#define INF 0x3f3f3f3f

#define maxn 1000+1



using namespace std;

struct Edge

{

    int from, to, dis;

    Edge(int u, int v, int w):from(u), to(v), dis(w)

    {}

};



struct HeapNode

{

    int d, u;

    bool operator <(const HeapNode&x)const

    {

        return d>x.d;

    }

};

struct Dijkstra

{

    int n, m;

    vector<Edge>edges;

    vector<int>G[maxn];

    bool vis[maxn]; //是否已永久标记

    int dis[maxn]; //s到各个点的距离

    int p[maxn]; //最短路中的上一个弧



    void init(int n)

    {

        this->n=n;

        for(int i=0; i<n; i++)

            G[i].clear();

        edges.clear();

    }

    void Add_edge(int from ,int to, int dis)

    {

        edges.push_back(Edge(from, to, dis));

        m=edges.size();

        G[from].push_back(m-1);

    }

    void dijkstra(int s)

    {

        priority_queue<HeapNode>q;

        for(int i=0; i<n; i++)

            dis[i]=INF;

        dis[s]=0;

        memset(vis, false, sizeof(vis));

        q.push(HeapNode{0, s} );

        while(!q.empty())

        {

            HeapNode x=q.top(); q.pop();

            int u=x.u;

            if(vis[u])

                continue;

            vis[u]=true;

            for(int i=0; i<G[u].size(); i++)

            {

                Edge &e=edges[G[u][i]];

                if(dis[e.to]>dis[u]+e.dis )

                {

                    dis[e.to]=dis[u]+e.dis;

                    p[e.to]=G[u][i];

                    q.push((HeapNode){dis[e.to], e.to} );

                }

            }

        }

    }

};

转载:http://blog.csdn.net/niushuai666/article/details/6791765

 2. Bellman-Ford算法总结:时间复杂度为:O(n*m)

 

Dijkstra算法是处理单源最短路径的有效算法,但它局限于边的权值非负的情况,若图中出现权值为负的边,Dijkstra算法就会失效,求出的最短路径就可能是错的。

这时候,就需要使用其他的算法来求解最短路径,Bellman-Ford算法就是其中最常用的一个。该算法由美国数学家理查德贝尔曼(Richard Bellman, 动态规划的提出者)和小莱斯特福特(Lester Ford)发明。

适用条件&范围:

 

单源最短路径(从源点s到其它所有顶点v);

有向图&无向图(无向图可以看作(u,v),(v,u)同属于边集E的有向图);

边权可正可负(如有负权回路输出错误提示);

差分约束系统;

 

Bellman-Ford算法的流程如下:
给定图G(V, E)(其中VE分别为图G的顶点集与边集),源点s,数组Distant[i]记录从源点s到顶点i的路径长度,初始化数组Distant[n], Distant[s]0

以下操作循环执行至多n-1次,n为顶点数:
对于每一条边e(u, v),如果Distant[u] + w(u, v) < Distant[v],则另Distant[v] = Distant[u]+w(u, v)w(u, v)为边e(u,v)的权值;
若上述操作没有对Distant进行更新,说明最短路径已经查找完毕,或者部分点不可达,跳出循环。否则执行下次循环;

为了检测图中是否存在负环路,即权值之和小于0的环路。对于每一条边e(u, v),如果存在Distant[u] + w(u, v) < Distant[v]的边,则图中存在负环路,即是说改图无法求出单源最短路径。否则数组Distant[n]中记录的就是源点s到各顶点的最短路径长度。

可知,Bellman-Ford算法寻找单源最短路径的时间复杂度为O(V*E).

BellmanFord算法可以大致分为三个部分
第一,初始化所有点。每一个点保存一个值,表示从原点到达这个点的距离,将原点的值设为0,其它的点的值设为无穷大(表示不可达)。
第二,进行循环,循环下标为从1n1n等于图中点的个数)。在循环内部,遍历所有的边,进行松弛计算。
第三,遍历途中所有的边(edgeuv)),判断是否存在这样情况:
dv) > d (u) + w(u,v)
则返回false,表示途中存在从源点可达的权为负的回路。
 
之所以需要第三部分的原因,是因为,如果存在从源点可达的权为负的回路。则 应为无法收敛而导致不能求出最短路径。 

测试代码如下:(下面为有向图的Bellman-Ford算法。。。。。)

#include<iostream>

#include<cstdio>

using namespace std;



#define MAX 0x3f3f3f3f

#define N 1010

int nodenum, edgenum, original; //点,边,起点



typedef struct Edge //边

{

	int u, v;

	int cost;

}Edge;



Edge edge[N];

int dis[N], pre[N];



bool Bellman_Ford()

{

	for(int i = 1; i <= nodenum; ++i) //初始化

		dis[i] = (i == original ? 0 : MAX);

	for(int i = 1; i <= nodenum - 1; ++i)

		for(int j = 1; j <= edgenum; ++j)

			if(dis[edge[j].v] > dis[edge[j].u] + edge[j].cost) //松弛(顺序一定不能反~)

			{

				dis[edge[j].v] = dis[edge[j].u] + edge[j].cost;

				pre[edge[j].v] = edge[j].u;

			}

			bool flag = 1; //判断是否含有负权回路

			for(int i = 1; i <= edgenum; ++i)

				if(dis[edge[i].v] > dis[edge[i].u] + edge[i].cost)

				{

					flag = 0;

					break;

				}

				return flag;

}



void print_path(int root) //打印最短路的路径(反向)

{

	while(root != pre[root]) //前驱

	{

		printf("%d-->", root);

		root = pre[root];

	}

	if(root == pre[root])

		printf("%d\n", root);

}



int main()

{

	scanf("%d%d%d", &nodenum, &edgenum, &original);

	pre[original] = original;

	for(int i = 1; i <= edgenum; ++i)

	{

		scanf("%d%d%d", &edge[i].u, &edge[i].v, &edge[i].cost);

	}

	if(Bellman_Ford())

		for(int i = 1; i <= nodenum; ++i) //每个点最短路

		{

			printf("%d\n", dis[i]);

			printf("Path:");

			print_path(i);

		}

	else

		printf("have negative circle\n");

	return 0;

}

 

测试数据:



4 6 1

1 2 20

1 3 5

4 1 -200

2 4 4

4 2 4

3 4 2



和:



4 6 1

1 2 2

1 3 5

4 1 10

2 4 4

4 2 4

3 4 2

 Bellman-Ford算法的核心代码模板:

bool Bellman_Ford()  

{  

    for(int i = 1; i <= nodenum; ++i) //初始化  

        dis[i] = (i == original ? 0 : MAX);  

    for(int i = 1; i <= nodenum - 1; ++i)  

        for(int j = 1; j <= edgenum; ++j)  

            if(dis[edge[j].v] > dis[edge[j].u] + edge[j].cost) //松弛(顺序一定不能反~)  

            {  

                dis[edge[j].v] = dis[edge[j].u] + edge[j].cost;              }  

            bool flag = 1; //判断是否含有负权回路  

            for(int i = 1; i <= edgenum; ++i)  

                if(dis[edge[i].v] > dis[edge[i].u] + edge[i].cost)  

                {  

                    flag = 0;  

                    break;  

                }  

                return flag;  

}  

 样例题目:hihocoder 1081   ( n个点, m条边,起点s,终点是t,求s到t的最短距离?)

 代码:

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <algorithm>

#define N 10000+5

#define INF 0x3f3f3f3f



struct Edge

{

    int u, v;

    int w;

}edge[N];

int n, m;



int dis[N];

int Bellman_Ford(int s, int e, int dd)

{

    int i, j;

    for(i=1; i<=n; i++)

    {

        dis[i]=(i==s?0:INF);

    }

    for(i=0; i<n-1; i++) //迭代n-1

    {

        for(j=0; j<dd; j++)

        {

            if( dis[ edge[j].v] > dis[edge[j].u]+edge[j].w )

                dis[ edge[j].v] = dis[edge[j].u]+edge[j].w; //路径松弛

        }

    }

    bool flag=true; //假设存在合法的最短路

    for(i=0; i<dd; i++)

    {

        if(dis[ edge[j].v] > dis[edge[j].u]+edge[j].w )//只有存在负环才会出现这种情况
{ flag=false; break; } } return flag; } int main() { int s, t; scanf("%d %d %d %d", &n, &m, &s, &t); int a, b, c; int e=0; for(int i=0; i<m; i++) { scanf("%d %d %d", &a, &b, &c ); edge[e].u=a; edge[e].v=b; edge[e++].w=c; edge[e].v=a; edge[e].u=b; edge[e++].w=c; //建立无向图 } if(Bellman_Ford(s, t, e)==true ) { printf("%d\n", dis[t] ); //输出起点到终点的最短距离 } else printf("......\n"); //如果不存在最短路就输出。。。。。。 return 0; }

Bellman_Ford算法还可统一用FIFO队列来处理,并且比较常用!

 

3. Floyd算法总结 :O(n^3)  建议数据量大概在:200--400左右

   算法:如果要求出图中每两点之间的最短路,不必调用n次Dijkstra(边权均为正),或者Bellman_Ford算法(有负权)。有个更简单的算法可以实现---Floyd-Warshall算法

  代码:

for(i=0; i<n; i++)

{

    for(j=0; j<n; j++)

        dis[i][j]=(i==j?0:INF); //初始化:自己到自己的距离为0

}                              //自己到其它节点的距离为无穷大

 

for(k=0; k<n; k++)

{

    for(i=0; i<n; i++)

    {

        for(j=0; j<n; j++)

            dis[i][j]=min(dis[i][j], dis[i][k]+dis[k][j] );

    }

}

 注意:这里会有一个潜在的问题:就是数据类型的溢出问题,如果INF定义太大,加法dis[i][k]+dis[k][j]可能会有溢出的危险。但是,如果定义太小则可能称为最短路的一部分。

 为此,估计一下最短路的实际长度上限,然后INF 设置成比它稍大一点的值。例如:最多有1000条边,边的权值最大不超过1000, 则可以定义INF=1000*1000+1

 但是还有更可靠的写法,代码:

for(k=0; k<n; k++)

{

    for(i=0; i<n; i++)

    {

        for(j=0; j<n; j++)

            if(dis[i][k]<INF && dis[k][j]<INF )

            {

                dis[i][j]=min(dis[i][j], dis[i][k]+dis[k][j] );

            }

    }

}

 

你可能感兴趣的:(最短路)