CF-Educational Round102-E-Minimum Path:最短路变形

题目大意:

给你一张无向带权图。询问从1点出发到每个点的最小
∑ i = 1 k e k i − max ⁡ i = 1 k e k i + min ⁡ i = 1 k e k i    ( e 1 , e 2 , . . . , e k 为 从 1 到 x 的 路 径 ( 不 一 定 是 简 单 路 径 ) 上 的 边 权 ) \sum_{i=1}^{k}e_{k_i}-\max_{i=1}^{k}e_{k_i}+\min_{i=1}^{k}e_{k_i} \ \ (e_1,e_2,...,e_k为从1到x的路径(不一定是简单路径)上的边权) i=1kekimaxi=1keki+mini=1keki  (e1,e2,...,ek1x())

n , m ≤ 2 e 5 , e i ≤ 1 e 9 n,m \leq 2e5,e_i \leq 1e9 n,m2e5,ei1e9

题目思路:

首先,若没有后面两个式子,那么就是一个最短路。所以可以预料到这是一个最短路的变形题.

考虑这样的图:
3 2
1 2 4
2 3 1

那么 d i s [ 2 ] = 3 dis[2]=3 dis[2]=3.因为我们可以这么走: 1 − > 2 − > 3 − > 2 1->2->3->2 1>2>3>2

这是可以来回走的模型,自然就想到要对图分层.(即对状态升维).

将这个问题转化为:跑最短路,你可以且必须进行如下两种操作:
1.选择路径上的一条边,减去它的边权
2.选择路径上的一条边,再加一次它的边权。

那么 d i s [ i ] [ 0 / 1 [ 0 / 1 ] dis[i][0/1[0/1] dis[i][0/1[0/1]代表从 1 1 1 i i i点,是否已经使用了操作1,是否已经使用了操作2的最小值.

那么答案就是 d i s [ i ] [ 1 ] [ 1 ] dis[i][1][1] dis[i][1][1].

正确性:很显然,对于任意一条路径,我们肯定是减去最大边权,加上最小边权。但问题是我们在跑最短路的过程中不知道哪条边会是最终的最小边权/最大边权(有后效性)。所以我们需要对状态升维来消除后效性.

心得:不管式子怎么复杂,其实我们发现它只多了两项,也就是只有两条边最终会受影响。所以可以考虑暴力升维,从而表示出所有状态。

AC代码:

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define pii pair<int,int>
#define pb push_back
#define mp make_pair
const int maxn = 5e5 + 5;
const int maxm = 5e5 + 5;
const int mod = 1e9 + 7;
int u[maxm] , v[maxm] , nextt[maxm] , first[maxn] , sign;
ll w[maxm];
void addedge(int x , int y , int z){
    u[++sign]=x;
    v[sign]=y;
    w[sign] = z;
    nextt[sign]=first[x];
    first[x]=sign;
}
struct Node{
    ll id , dis;
    int s0 , s1;
    Node (int n = 0, ll d = 1e17 , int s0 = 0 , int s1 = 0)
    {
        id = n;dis = d;
        this->s0 = s0;
        this->s1 = s1;
    }
    bool operator < (const Node & a)const
    {
        return dis > a.dis;
    }
};
int book[maxn][2][2] , n , m;
ll dis[maxn][2][2];
void Dijstra(ll s){
    priority_queue<Node>q;
    while(q.size()) q.pop();
    for (int i = 1 ; i <= n ; i++){
        for (int j = 0 ; j <= 1 ; j++){
            for (int k = 0 ; k <= 1 ; k++){
                book[i][j][k] = 0;
                dis[i][j][k] = 1e16;
            }
        }
    }
    dis[s][0][0] = 0;
    q.push(Node(s , 0 , 0 , 0));
    while(q.size())
    {
        Node g = q.top() ; q.pop();
        int id = g.id;
        ll dist = g.dis;
        int s0 = g.s0 , s1 = g.s1;
        if (book[id][s0][s1]) continue;
        book[id][s0][s1] = 1;
        for (int i = first[id] ; i ; i = nextt[i]){
            if (dis[v[i]][s0][s1] > dist + w[i]){
                dis[v[i]][s0][s1] = dist + w[i];
                q.push(Node(v[i] , dis[v[i]][s0][s1] , s0 , s1));
            }
            if (!s0 && dis[v[i]][1][s1] > dist){
                dis[v[i]][1][s1] = dist;
                q.push(Node(v[i] , dis[v[i]][1][s1] , 1 , s1));
            }
            if (!s1 && dis[v[i]][s0][1] > dist + 2 * w[i]){
                dis[v[i]][s0][1] = dist + 2 * w[i];
                q.push(Node(v[i] , dis[v[i]][s0][1] , s0 , 1));
            }
            if (!s0 && !s1 && dis[v[i]][1][1] > dist + w[i]){
                dis[v[i]][1][1] = dist + w[i];
                q.push(Node(v[i] , dis[v[i]][1][1] , 1 , 1));
            }
        }
    }
    return ;
}
int main()
{
    ios::sync_with_stdio(false);
    cin >> n >> m;
    for (int i = 1 ; i <= m ; i++){
        int x , y , z; cin >> x >> y >> z;
        addedge(x , y , z);
        addedge(y , x , z);
    }
    Dijstra(1);
    for (int i = 2 ; i <= n ; i++){
        cout << dis[i][1][1] << " ";
    }
    cout << endl;
    return 0;
}
  • 7
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值