吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 1636|回复: 6
收起左侧

[C&C++ 原创] 单源最短路算法:Dijkstra 讲解

[复制链接]
RainPPR 发表于 2022-11-27 15:19
本帖最后由 RainPPR 于 2022-11-27 15:28 编辑

单源最短路算法:Dijkstra

适用场景

单源最短路:从1号点到其他所有点的最短距离
没有负权变:所有边的权重(长度)都是正数

朴素版 Dijkstra 适用于【稠密图】

原题链接

特征

  • 贪心
  • 时间复杂度:O(n^2)

思路

  1. 初始化距离:dist[1] = 0, dist[i] = ∞;
    1 号点到 1 号点的距离是 0;其他点到 1 号点的距离设为正无穷
  2. 集合 s:存储已经确定最短距离的点
  3. 循环:for i : 0 ~ n
    1. t ← 不在 s 中的,距离最短的点
    2. 将 t 加入 s 中:s ← t
    3. 用 t 更新其他所有点的距离(从 t 出去的所有边,能不能更新其他点的距离):

C++代码

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>

using namespace std;

const int N = 510;
const int INF = 0x3f3f3f3f;

// 返回最大值
inline int MAX(int a, int b)
{
        return (a > b) ? a : b;
}
// 返回最小值
inline int MIN(int a, int b)
{
        return (a < b) ? a : b;
}

int n, m;                // n:点数 m:边数
int g[N][N];        // 稠密图用邻接矩阵存储
int dist[N];        // 从1号点走到每个点的最短距离
bool st[N];                // 这个点的最短距离是否已经确定

// 求出 1 号点到 n 号点的最短距离
int dijkstra()
{
        // 初始化距离
        memset(dist, 0x3f, sizeof(dist));
        dist[1] = 0;

        // 循环 n 次
        for (int i = 0 ; i < n ; ++i)
        {
                int t = -1;        // 不在 s 中的,距离最短的点
                for (int j = 1 ; j <= n ; ++j)
                        if (!st[j] && (t == -1 || dist[t] > dist[j]))
                                t = j;
                st[t] = true;

                // 如果已经找到最短路:退出
                if (t == n)
                        break;

                // 用 t 更新其他所有点的距离
                for (int j = 1 ; j <= n ; ++j)
                        dist[j] = MIN(dist[j], dist[t] + g[t][j]);
        }

        // 如果没有路径
        if (dist[n] == INF)
                // 返回 -1
                return -1;
        // 返回 1 号点到 n 号点的距离
        return dist[n];
}

int main()
{
        // 初始化邻接矩阵
        memset(g, 0x3f, sizeof(g));

        // 输入
        scanf("%d %d", &n, &m);
        int a, b, c;
        while (m--)
        {
                scanf("%d %d %d", &a, &b, &c);
                g[a][b] = MIN(g[a][b], c);        // 图中可能存在重边和自环:保留长度最短的边
        }

        // 朴素版Dijkstra
        int t = dijkstra();

        // 输出
        printf("%d\n", t);

        return 0;
}

堆优化 Dijkstra 适用于【稀疏图】

原题链接

特征

  • 贪心、堆
  • 时间复杂度:O(m log n)

回顾朴素版的思路

  1. 初始化距离:dist[1] = 0, dist[i] = ∞;
    1 号点到 1 号点的距离是 0;其他点到 1 号点的距离设为正无穷
  2. 集合 s:存储已经确定最短距离的点
  3. 循环:for i : 0 ~ n
    1. t ← 不在 s 中的,距离最短的点
    2. 将 t 加入 s 中:s ← t
    3. 用 t 更新其他所有点的距离(从 t 出去的所有边,能不能更新其他点的距离):
通过分析可知,3.1 找距离最短的点复杂度高,可使用“堆(SLT-优先队列)”进行维护
缺点:STL不能修改任意元素,可能有冗余

C++代码

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<queue>

using namespace std;

typedef pair<int, int> PII;

const int N = 1e6 + 10;
const int INF = 0x3f3f3f3f;

// 返回最大值
inline int MAX(int a, int b)
{
        return (a > b) ? a : b;
}
// 返回最小值
inline int MIN(int a, int b)
{
        return (a < b) ? a : b;
}

int n, m;                // n:点数 m:边数

int h[N], w[N], e[N], ne[N], idx;        // 稀疏图用邻接表存储
int dist[N];        // 从1号点走到每个点的最短距离
bool st[N];                // 这个点的最短距离是否已经确定

void add(int a, int b, int c)
{
        e[idx] = b;
        w[idx] = c;
        ne[idx] = h[a];
        h[a] = idx++;
}

// 求出 1 号点到 n 号点的最短距离
int dijkstra()
{
        // 初始化距离
        memset(dist, 0x3f, sizeof(dist));
        dist[1] = 0;

        // 用堆维护、查找距离最短的点
        priority_queue<PII, vector<PII>, greater<PII>> heap;
        heap.push({0, 1});

        while (heap.size())
        {
                // 不在 s 中的,距离最短的点
                PII t = heap.top();
                heap.pop();

                int ver = t.second, distance = t.first;
                if (st[ver])        // 确保不在 s 中
                        continue;
                st[ver] = true;

                // 用 t 更新其他所有点的距离
                for (int i = h[ver] ; i != -1 ; i = ne[i])
                {
                        int j = e[i];
                        if (dist[j] > distance + w[i])
                        {
                                dist[j] = distance + w[i];
                                heap.push({dist[j], j});
                        }
                }
        }

        // 如果没有路径
        if (dist[n] == INF)
                // 返回 -1
                return -1;
        // 返回 1 号点到 n 号点的距离
        return dist[n];
}

int main()
{
        // 初始化邻接表
        memset(h, -1, sizeof(h));

        // 输入
        scanf("%d %d", &n, &m);
        int a, b, c;
        while (m--)
        {
                scanf("%d %d %d", &a, &b, &c);
                add(a, b, c);        // 用邻接表存储,无需判断重边
        }

        // 堆优化Dijkstra
        int t = dijkstra();

        // 输出
        printf("%d\n", t);

        return 0;
}


新人第一次发帖,如有问题欢迎大佬指出,谢谢
本文Markdown源码: Dijkstra.zip (4.13 KB, 下载次数: 9)

免费评分

参与人数 1吾爱币 +5 热心值 +1 收起 理由
wushaominkk + 5 + 1 用心讨论,共获提升!

查看全部评分

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

zws9812 发表于 2022-11-28 20:51
感谢分享谢谢了
my52pojie110 发表于 2022-11-28 22:31
Penguinbupt 发表于 2022-11-30 14:12
Bowyn09 发表于 2022-11-30 15:23

楼主威武,感谢分享
525308a 发表于 2022-11-30 15:41
朴素而又经典的算法
lfordch 发表于 2022-12-3 02:11
感谢分享,学习了
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2025-1-11 23:36

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表