登录后台

页面导航

本文编写于 1353 天前,最后修改于 1353 天前,其中某些信息可能已经过时。

题目链接:

https://pintia.cn/problem-sets/994805046380707840/problems/1386335159927652364

题解:

建两个图,正图和反图。

正图里的边权是现金,反图里的边权是旅游金。

然后分别以1为起点在正图上跑最短路,以n为起点在反图上跑最短路。

这样计算出每个点的答案,取最小,不带修改的情况就做完了。

带修改的情况放线段树上维护一下就好了。

注意巨大坑点:不保证图连通。

参考链接:

https://www.cnblogs.com/zhanglichen/p/14726600.html

代码:

#pragma GCC optimize(1)
#pragma GCC optimize(2)
#pragma GCC optimize(3,"Ofast","inline")
#include <bits/stdc++.h>
#define inf 0x3f3f3f3f3f3f3f
using namespace std;
typedef long long ll;

struct node
{
    ll to,val;
};
vector<node>v_mon[200500],v_bi[200500];
ll a[200500]= {0};


struct node1
{
    ll now,val;
    bool operator <(const node1& a) const
    {
        return val>a.val;
    }
};
priority_queue<node1>que1,que2;
ll dis_mon[200500]= {0},dis_bi[200500]= {0};


ll tree[800500]= {0};
ll min1[800500]= {0};
ll val[800500]= {0};
void push_up(ll n)
{
    min1[n]=min(min1[2*n],min1[2*n+1]);
}
void build(ll n,ll l,ll r)
{
    if(l==r)
    {
        min1[n]=tree[n]=val[l];
        return ;
    }
    ll mid=(l+r)/2;
    build(2*n,l,mid);
    build(2*n+1,mid+1,r);
    push_up(n);
}
void update(ll n,ll L,ll R,ll pos,ll val1)
{
    if(L==R&&L==pos)
    {
        min1[n]=tree[n]=val1;
        return ;
    }
    ll mid=(L+R)/2;
    if(pos<=mid)
        update(2*n,L,mid,pos,val1);
    else
        update(2*n+1,mid+1,R,pos,val1);
    push_up(n);
}
int main()
{
    ll n,m,k,from,to,val1,val2,pos;
    scanf("%lld%lld%lld",&n,&m,&k);

    for(ll i=1; i<=n; i++)
        dis_mon[i]=inf,dis_bi[i]=inf;

    dis_mon[1]=0,dis_bi[n]=0;
    for(ll i=1; i<=m; i++)
    {
        scanf("%lld%lld%lld%lld",&from,&to,&val1,&val2);
        v_mon[from].push_back({to,val1});
        v_bi[to].push_back({from,val2});
    }

    que1.push({1,0});
    while(!que1.empty())
    {
        ll now=que1.top().now,val=que1.top().val;
        que1.pop();
        for(node i:v_mon[now])
        {
            if(dis_mon[i.to]>val+i.val)
            {
                dis_mon[i.to]=val+i.val;
                que1.push({i.to,dis_mon[i.to]});
            }
        }
    }

    que2.push({n,0});
    while(!que2.empty())
    {
        ll now=que2.top().now,val=que2.top().val;
        que2.pop();
        for(node i:v_bi[now])
        {
            if(dis_bi[i.to]>val+i.val)
            {
                dis_bi[i.to]=val+i.val;
                que2.push({i.to,dis_bi[i.to]});
            }
        }
    }

    for(ll i=1; i<=n; i++)
        scanf("%lld",&a[i]);

    for(ll i=1; i<=n; i++)
    {
        if(dis_mon[i]==inf||dis_bi[i]==inf)
            val[i]=inf;
        else
            val[i]=dis_mon[i]+dis_bi[i]/a[i]+(ll)((dis_bi[i]%a[i])>0?1:0);
    }

    build(1,1,n);

    for(ll i=1; i<=k; i++)
    {
        scanf("%lld",&pos);
        scanf("%lld",&a[pos]);
        if(dis_mon[pos]==inf||dis_bi[pos]==inf)
            val[pos]=inf;
        else
            val[pos]=dis_mon[pos]+dis_bi[pos]/a[pos]+(ll)((dis_bi[pos]%a[pos])>0?1:0);
        update(1,1,n,pos,val[pos]);
        printf("%lld\n",min1[1]);
    }
}