Dijkstra算法适合求不包含负权路的最短路径,通过点增广、在稠密图中使用优化过的版本速度非常可观。本篇不介绍算法原理、只给出模板,这里给出三种模板,其中最实用的是加上了堆优化的版本

算法原理 or 学习参考链接 : 点我 、不要点它点我!为何不适用于带负权边图

( Dijkstra 动态演示 )

朴素版 ( 邻接矩阵存储、复杂度 O( n2 ) )

///HDU 2544为例
#include<stdio.h>
#include<string.h>
const int INF  = 0x3f3f3f3f;
;

bool vis[maxn];
int G[maxn][maxn],dis[maxn],pre[maxn];//pre[]记录前驱、用于输出路径
int n, m;
void dijkstra(int v)
{
    int i, j, u , Min;
    ;i<=n;i++){
        dis[i]=G[v][i];
        vis[i]=;
        //if(i!=v&&G[v][i]!=INF)pre[i] = v;
       // else pre[i] = -1;
    }
    vis[v]=;dis[v]=;
    ;i<n;i++){
        Min = INF;
        ;j<=n;j++){
            if(!vis[j]&&Min > dis[j]){
                Min = dis[j];
                u = j;
            }
        }
        if(Min == INF)break;
        vis[u]=;
        ;j<=n;j++){
            if(!vis[j]&&G[u][j]!=INF&&dis[u]+G[u][j]<dis[j]){
                dis[j] = G[u][j] + dis[u];
              //  pre[j] = u;
            }
        }
    }
}
int main()
{
    int i, j, x, y, w;
    while(~scanf("%d%d",&n,&m)&&n)
    {
        ;i<=n;i++)
            ;j<=n;j++)
                ;
                else G[i][j] = INF;

        while(m--){
            scanf("%d%d%d",&x,&y,&w);
            G[x][y] = w;
            G[y][x] = w;
        }
        dijkstra();
        printf("%d\n",dis[n]);  //以下为输出路径
        /*int p, len=0, ans[maxn];
        p = n-1;
        while(p!=0)
        {
            ans[len++] = p;
            p = pre[p];

        }
        printf("0->");
        for(i=len-1;i>=0;i--)
            printf("%d",ans[i]);
        puts("");  */
    }
    ;
}

STL优先队列优化版本 ( 复杂度 O( (V+E)logV ) )、此优化需要用邻接表存图

///POJ 2387为例
#include<stdio.h>
#include<string.h>
#include<queue>
#include<algorithm>
#include<stdlib.h>
using namespace std;
;
const int INF = 0x3f3f3f3f;
typedef pair<int, int> HeapNode;///在堆里面的是pair、first为到起点距离、second为点编号
struct EDGE{ int v, nxt, w; };

int Head[maxn], Dis[maxn];
EDGE Edge[maxn*];
int N, M, cnt;

inline void init()
{
    ; i<=N; i++)
        Head[i]=-, Dis[i]=INF;
    cnt = ;
}

inline void AddEdge(int from, int to, int weight)
{
    Edge[cnt].w = weight;
    Edge[cnt].v = to;
    Edge[cnt].nxt = Head[from];
    Head[from] = cnt++;
}

int Dijkstra()
{
    priority_queue<HeapNode, vector<HeapNode>, greater<HeapNode> > Heap;
    Dis[] = ;
    Heap.push(make_pair(, ));
    while(!Heap.empty()){
        pair<int, int> T = Heap.top(); Heap.pop();
        if(T.first != Dis[T.second]) continue;///有很多版本都是用 vis 标记是否已经使用这个点松弛过、这里可以用这个不同的方法!

        ; i=Edge[i].nxt){
            int Eiv = Edge[i].v;
            if(Dis[Eiv] > Dis[T.second] + Edge[i].w){
                Dis[Eiv] = Dis[T.second] + Edge[i].w;
                Heap.push(make_pair(Dis[Eiv], Eiv));
            }
        }
    }
    return Dis[N];
}

int main(void)
{
    while(~scanf("%d %d", &M, &N)){

        init();

        int from, to, weight;
        ; i<M; i++){
            scanf("%d %d %d", &from, &to, &weight);
            AddEdge(from, to, weight);
            AddEdge(to, from, weight);
        }

        printf("%d\n", Dijkstra());
    }
    ;
}

传说中还有一种斐波那契堆,比STL默认的堆更高效、但是斐波那契堆难写难理解、故用配对堆来代替( 复杂度 O(VlogV + E) )

///POJ 2387为例
#include<stdio.h>
#include<algorithm>
#include<stdlib.h>
#include<string.h>
using namespace std;
;
const int INF  = 0x3f3f3f3f;
struct EDGE{ int v, nxt, w; };
EDGE Edge[maxn*maxn];
int Head[maxn], Dis[maxn], T, N, cnt;
int Cost[maxn][maxn];
inline void init()
{
    ; i<=N; i++){
        Head[i]=-,Dis[i]=INF;
        ; j<=N; j++){
            Cost[i][j] = INF;
        }
    }
    cnt=;
}

inline void ADD(int from, int to, int weight)
{
    Edge[cnt].w=weight,
    Edge[cnt].v = to;
    Edge[cnt].nxt = Head[from];
    Head[from] = cnt++;
}

struct Heap{
    int num[maxn],pos[maxn],Size;

    void PushUp(int p) {
        ) {
            ]]) {
                swap(num[p],num[p >> ]);
                swap(pos[num[p]],pos[num[p >> ]]);
                p >>= ;
            }
            else break;
        }
    }
    void Insert(long long x) {
        num[++Size] = x;
        pos[x] = Size;
        PushUp(Size);
    }
    void Pop() {
        pos[num[]] = ;
        num[] = num[Size--];
        ]] = ;
        ;
        while(now < Size) {
            ]] < Dis[num[now]])
                ++now;
            ]]) {
                swap(num[now],num[now >> ]);
                swap(pos[num[now]],pos[num[now >> ]]);
                now <<= ;
            }
            else break;
        }
    }
}heap;///配对堆

int Dijkstra()
{
    Dis[] = ;
    ; i<=N; i++) heap.Insert(i);
    while(heap.Size){
        ]; heap.Pop();
        ; i=Edge[i].nxt)
            if(Dis[Edge[i].v] > Dis[x] + Edge[i].w)
                Dis[Edge[i].v] = Dis[x] + Edge[i].w,
                heap.PushUp(heap.pos[Edge[i].v]);
    }
    return Dis[N];
}

int main(void)
{
    while(~scanf("%d %d", &T, &N)){

        init();

        int from, to, weight;
        ; i<T; i++){
            scanf("%d %d %d", &from, &to, &weight);
            if(Cost[from][to] > weight){
                Cost[from][to] = Cost[to][from] = weight;
                ADD(from, to, weight);
                ADD(to, from, weight);
            }
        }

        printf("%d\n", Dijkstra());
    }
    ;
}

手撕配对堆版本

在 Linux 下有pbds可以调用,里面可以调用二叉堆、配对堆、斐波那契堆……

///POJ 2387为例
#include<stdio.h>
#include<string.h>
#include<queue>
#include<algorithm>
#include<ext/pb_ds/priority_queue.hpp>///记得加上
#include<stdlib.h>
using namespace __gnu_pbds;///记得加上
using namespace std;
;
const int INF = 0x3f3f3f3f;
typedef pair<int, int> HeapNode;
struct EDGE{ int v, nxt, w; };

inline int read()
{
    ,f=;char ch=getchar();
    ;ch=getchar();}
    +ch-';ch=getchar();}
    return x*f;
}

int Head[maxn], Dis[maxn];
EDGE Edge[maxn*];
int N, M, cnt;

inline void init()
{
    ; i<=N; i++)
        Head[i]=-, Dis[i]=INF;
    cnt = ;
}

inline void AddEdge(int from, int to, int weight)
{
    Edge[cnt].w = weight;
    Edge[cnt].v = to;
    Edge[cnt].nxt = Head[from];
    Head[from] = cnt++;
}

int Dijkstra()
{
    __gnu_pbds::priority_queue<HeapNode,greater<HeapNode>,pairing_heap_tag > Heap;///申请方式、其余和普通优先队列无差别
    Dis[] = ;
    Heap.push(make_pair(, ));
    while(!Heap.empty()){
        pair<int, int> Top = Heap.top();
        Heap.pop();
        int v = Top.second;
        if(Top.first != Dis[v]) continue;
        ; i=Edge[i].nxt){
            int tmp = Edge[i].v;
            if(Dis[tmp] > Dis[v] + Edge[i].w){
                Dis[tmp] = Dis[v] + Edge[i].w;
                Heap.push(make_pair(Dis[tmp], tmp));
            }
        }
    }
    return Dis[N];
}

int main(void)
{
    while(~scanf("%d %d", &M, &N)){

        init();

        int from, to, weight;
        ; i<M; i++){
            from = read(); to = read(); weight = read();
            AddEdge(from, to, weight);
            AddEdge(to, from, weight);
        }

        printf("%d\n", Dijkstra());
    }
    ;
}

pbds

Dijkstra算法求最短路模板的更多相关文章

  1. 2019中山纪念中学夏令营-Day14 图论初步【dijkstra算法求最短路】

    Dijkstra是我学会的第一个最短路算法,为什么不先去学SPFA呢?因为我在luogu上翻到了一张比较神奇的图: 关于SPFA -它死了 以及网上还有各位大佬的经验告诉我:SPFA这玩意很容易被卡. ...

  2. dijkstra算法求最短路

    艾兹格·W·迪科斯彻 (Edsger Wybe Dijkstra,1930年5月11日~2002年8月6日)荷兰人. 计算机科学家,毕业就职于荷兰Leiden大学,早年钻研物理及数学,而后转为计算学. ...

  3. A*算法求K短路模板 POJ 2449

    #include<cstdio> #include<queue> #include<cstring> using namespace std; const int ...

  4. 关于dijkstra求最短路(模板)

    嗯....   dijkstra是求最短路的一种算法(废话,思维含量较低,   并且时间复杂度较为稳定,为O(n^2),   但是注意:!!!!         不能处理边权为负的情况(但SPFA可以 ...

  5. Dijkstra算法求单源最短路径

    Description 在每年的校赛里,所有进入决赛的同学都会获得一件很漂亮的t-shirt.但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以现在他们想要寻找最短的从商店 ...

  6. Dijkstra算法求最短路径(java)(转)

    原文链接:Dijkstra算法求最短路径(java) 任务描述:在一个无向图中,获取起始节点到所有其他节点的最短路径描述 Dijkstra(迪杰斯特拉)算法是典型的最短路径路由算法,用于计算一个节点到 ...

  7. _DataStructure_C_Impl:Dijkstra算法求最短路径

    // _DataStructure_C_Impl:Dijkstra #include<stdio.h> #include<stdlib.h> #include<strin ...

  8. 【POJ - 2139】Six Degrees of Cowvin Bacon (Floyd算法求最短路)

    Six Degrees of Cowvin Bacon Descriptions 数学课上,WNJXYK忽然发现人缘也是可以被量化的,我们用一个人到其他所有人的平均距离来量化计算. 在这里定义人与人的 ...

  9. 《算法导论》读书笔记之图论算法—Dijkstra 算法求最短路径

    自从打ACM以来也算是用Dijkstra算法来求最短路径了好久,现在就写一篇博客来介绍一下这个算法吧 :) Dijkstra(迪杰斯特拉)算法是典型的最短路径路由算法,用于计算一个节点到其他所有节点的 ...

随机推荐

  1. python3 基本数据类型_2

    #!/usr/bin/python3 #以下set,dict的方法py2无法运行 #创建set 集合,使用大括号 { } 或者 set() 函数创建 #注意:创建一个空集合必须用 set() 而不是 ...

  2. charles抓包教程

    百度搜索下载charles 默认安装即可完成 1.双击charles.exe启动,我的是4.2.7版本.最好下载原版的不要去破解中文,会有不兼容 1.搜索该软件许可证书并输入即可长期使用 2.设置代理 ...

  3. diff patch比较文件打补丁

    比较文件将结果保存到patch文件:diff -u test1.txt test2.txt > patchfile test1.txt应用patch文件,并备份(test1.txt.orig): ...

  4. Go语言入门篇-Golang之文本编码处理

    Golang之文本编码处理

  5. 63 (OC)* NSAutoreleasePool 自动释放池

    目录 0:ARC 1: 自动释放池 2:NSAutoreleasePool实现原理 3:autorelease 方法 4: Runloop和Autorelease的关系 5: Using Autore ...

  6. (4.25)Sqlserver中 登录用户只能看到自己拥有权限的库

    Sqlserver中 登录用户只能看到自己拥有权限的库 转自:https://www.cnblogs.com/huangtailang/p/4209180.html 相关参考:https://www. ...

  7. iScroll使用参考

    分享是传播.学习知识最好的方法 以下这篇文章是iScroll.js官网的中文翻译,尽管自己英文不好,但觉得原作者们翻译的这个资料还是可以的,基本用法介绍清楚了.如果你英文比较好的话,可以看看官网的资料 ...

  8. Windows 运行时加载动态库

    下面是一个运行时加载nvcuda.dll,并检测当前驱动版本最大支持的CUDA版本的例子. #include "cuda.h" #include <stdio.h> # ...

  9. [2019多校联考(Round 6 T3)]脱单计划 (费用流)

    [2019多校联考(Round 6 T3)]脱单计划 (费用流) 题面 你是一家相亲机构的策划总监,在一次相亲活动中,有 n 个小区的若干男士和 n个小区的若干女士报名了这次活动,你需要将这些参与者两 ...

  10. Codeforces 1262D Optimal Subsequences(BIT+二分)

    首先比较容易想到肯定是前k大的元素,那么我们可以先对其进行sort,如果数值一样返回下标小的(见题意),接下里处理的时候我们发现需要将一个元素下标插入到有序序列并且需要访问第几个元素是什么,那么我们可 ...