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

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队列来处理,并且比较常用!

 

 SPFA算法(Bellman-ford的队列优化),只要最短路径存在,SPFA算法必能求出最小值。

题目:hihocoder http://hihocoder.com/problemset/problem/1093

样例输入
5 10 3 5
1 2 997
2 3 505
3 4 118
4 5 54
3 5 480
3 4 796
5 2 794
2 5 146
5 4 604
2 5 63
样例输出
172

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#include <math.h>
#include <iostream>
#include <string>
#include <queue>
#include <vector>
#include <algorithm>
#define INF 0x3f3f3f3f

using namespace std;
int n, m, st, en;
struct node
{
    int v, w;
};
vector<node>q[100000+100];
int dis[100000+100];
bool vis[100000+100];

void SPFA()
{
    queue<int>que;
    memset(vis, 0, sizeof(vis));
    for(int i=0; i<=n; i++)
        dis[i]=INF;
    dis[st]=0;
    que.push(st);
    while(!que.empty())
    {
        int cur=que.front(); que.pop();

        vis[cur]=1;
        int len=q[cur].size();
        for(int i=0; i<len; i++){
            int v=q[cur][i].v; //v是与之相连的点
            if(dis[v]>(dis[cur]+q[cur][i].w) ){//如果当前保存的到v点的距离 大于 从cur到v间接路径的距离
                dis[v] = dis[cur]+q[cur][i].w;//进行一次松弛
                if(!vis[v]){
                    que.push(v); vis[v]=1;//入队列标记访问
                }
            }
        }
        vis[cur]=0;//将该点的标记撤销
    }
}

int main()
{
    scanf("%d %d %d %d", &n, &m, &st, &en);
    int u, v, w;
    node temp;
    for(int i=0; i<m; i++){
        scanf("%d %d %d", &u, &v, &w);
        temp.v=v; temp.w=w;
        q[u].push_back(temp);
        temp.v=u;
        q[v].push_back(temp);
    }//建立双向边
    SPFA();
    printf("%d\n", dis[en]);
    return 0;
}

 

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] );
            }
    }
}

 

posted @ 2015-02-04 14:46  我喜欢旅行  阅读(313)  评论(0编辑  收藏  举报