Dijkstra求最短路Ⅱ|堆优化 稀疏矩阵优化

Dijkstra求最短路Ⅱ|堆优化 稀疏矩阵优化


from acwing850
Time limit:1s
Memory limit:64MB

Problem Description

给定一个n个点m条边的有向图,图中可能存在重边和自环,所有边权均为非负值。

请你求出1号点到n号点的最短距离,如果无法从1号点走到n号点,则输出-1。

Input

第一行包含整数n和m。

接下来m行每行包含三个整数x,y,z,表示存在一条从点x到点y的有向边,边长为z。

Output

输出一个整数,表示1号点到n号点的最短距离。

如果路径不存在,则输出-1。

数据范围

1≤n,m≤1.5×105 ,1≤n,m≤1.5×105,
图中涉及边长均不小于0,且不超过10000。

Sample Input
3 3
1 2 2
2 3 1
1 3 4
Sample Output
3

对于这个题目,和Dijkstra求最短路Ⅰ相比,数据量变大了,且点边的量级也改变了,对于一般最短路问题,边是点量级的平方,那么一般数据量都很小,可以直接开辟一个二维数组;但是对于大一点的数据量,边和点是相同量级的时候,内存是不够的开辟的,那么我们就要用到稀疏矩阵,或者vector模拟链表(指针)也可以。

(这里要自闭一下,这学期刚学习了稀疏矩阵,结果差点都忘了,还想了好一会儿,不过这个题目也就用到了存储而已,然后加了一点并查集的思想用来索引)

当然,传统朴素dijkstra算法的时候复杂度是O(n^2)的,这里我们利用优先队列降低时间复杂度,并且按照权值从小到大排序(greater),这样每次弹出的元素都是未被标记的点中,离1(起点)最近的,这个操作相当于dijkstra朴素方法中遍历了1到n找最小距离点。

AC代码:
#include<cstdio>
#include<queue>
#include<cstring>
#define pii pair<int,int>
using namespace std;
int n,m;
int from[150005],father[150005],dis[150005],edge = 0;pii p[150005];
bool note[150005];
void init(){
    scanf("%d %d",&n,&m);
    memset(from,-1,sizeof(from));   //后续操作当遇到-1的时候结束dijsktra内部循环
    int f,t,w;
    while(m--){
        scanf("%d %d %d",&f,&t,&w);
        p[edge].first = w,p[edge].second = t,father[edge] = from[f],from[f] = edge++;
    }//这个步骤,是用pair存输入的边的终点(second)和边权(first)
     //father可以理解为并查集,对于所有起点相同的边,下一个位置,存着上一个的索引,而from是用来索引最后一个	   //father的,这里可以和并查集的查找函数联系起来理解
    for(int i = 2;i <= n;++i)
        dis[i] = 0x3f3f3f3f;
}
int main(){
    init();
    priority_queue<pii,vector<pii>,greater<pii> > q;
    q.push({0,1});          //优先队列,按边权从小到大排序,所以first存的是边权
    while(!q.empty()){
        pii x = q.top();q.pop();
        if(note[x.second])
            continue;
        note[x.second] = true;
        for(int i = from[x.second];i != -1;i = father[i])   //经典并查集思想了
            if(!note[p[i].second] && dis[p[i].second] > dis[x.second] + p[i].first)
                dis[p[i].second] = dis[x.second] + p[i].first,q.push({dis[p[i].second],p[i].second});//更新了就压入队列
    }
    if(dis[n] == 0x3f3f3f3f)
        printf("-1");
    else
        printf("%d",dis[n]);
    return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Alan_Lowe

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值