zoukankan      html  css  js  c++  java
  • 最短路算法总结(*【模板】)

    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
    ", 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
    ", 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
    ", dis[i]);
    			printf("Path:");
    			print_path(i);
    		}
    	else
    		printf("have negative circle
    ");
    	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 ", dis[t] ); //输出起点到终点的最短距离 } else printf("...... "); //如果不存在最短路就输出。。。。。。 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
    ", 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] );
                }
        }
    }
    
  • 相关阅读:
    HashMap 链表插入方式 → 头插为何改成尾插 ?
    MySQL 日志之 binlog 格式 → 关于 MySQL 默认隔离级别的探讨
    Eclipse
    Delphi
    Delphi
    Delphi
    Delphi
    Delphi
    Delphi
    Delphi
  • 原文地址:https://www.cnblogs.com/yspworld/p/4272337.html
Copyright © 2011-2022 走看看