分层建图+最短路

前言:分层图不仅需要知道这个概念,还需要我们巧妙的建图

这个题目需要建立一个中转的虚层

思路: 这就是建分层图的第一种情况,有k条地铁线路,我们就建k张图,然后再建第k+1张虚层,将各个可以中转线路的点连接起来,并设定从虚层到地铁需要乘该线的代价,从地铁到虚层代价为0,这里的虚层就相当于地铁站,这就实现了地铁的转线。


题目地址

在这里插入图片描述

#include<bits/stdc++.h>
#include<iostream>
using namespace std;

const int N = (int)5e5+10;
int n,m,s,t;
int e[N],ne[N],h[N],idx = 0;
int w[N];

int vis[N];
int dis[N];

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

void dij(int s){
    memset(dis,0x3f3f3f3f,sizeof dis);
    dis[s] = 0;
    priority_queue<pair<int,int>,vector<pair<int,int>>,greater<pair<int,int>>> q;
    q.push({0,s});
    while(q.size()){
        int d = q.top().first, u = q.top().second;
        q.pop();
        if(vis[u]) continue; vis[u] = 1;
        for(int i=h[u];i;i=ne[i]){
            int to = e[i], wei = w[i];
            if(dis[u]+wei<dis[to]){
                dis[to] = dis[u] + wei;
                q.push({dis[to],to});
            }
        }
    }
}

signed main(){
    cin >> n >> m >> s >> t;
    for(int i=1;i<=m;i++){
        int a,b,c; cin >> a >> b >> c;
        int pre;
        for(int j=0;j<c;j++){
            int cur; cin >> cur;
            if(j){
                add((i-1)*n+pre,(i-1)*n+cur,b);
                add((i-1)*n+cur,(i-1)*n+pre,b);
            }
            add((i-1)*n+cur,n*m+cur,0);
            add(n*m+cur,(i-1)*n+cur,a);
            pre = cur;
        }
    }
    dij(n*m+s);
    if(dis[n*m+t]==0x3f3f3f3f) cout << -1;
    else cout << dis[n*m+t];
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

wniuniu_

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

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

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

打赏作者

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

抵扣说明:

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

余额充值