loj 2759「JOI 2014 Final」飞天鼠

loj

这题有在一棵树上上升或者下降的操作,稍加分析后可以发现上升操作如果不是一定要做(指高度不足以到下一棵树或者是最后到达\(n\))就不做,下降操作也是如果不是一定要做(指到达下一棵树时高度过高)就不做,因为如果提前做了,可能会导致后面要浪费一些步数使得移动合法.然后这个移动过程就会分成两段,先是一直移动或者下降,不用上升,然后会每次上升再移动,一直到终点

先看前一段的移动,如果移动的时候正好能移到下一棵树就直接移,如果移的时候高度过高就往下移一点直到能正好移动到下一棵树上.这里对每个点记\(di_x\)表示从起点到这里用了多久.注意我们还需要知道在某个树上所在的高度,由于这一部分没有上升操作,并且每过一个单位时间会下降1,所以在某点高度为\(X-di_x\).然后这个\(di_x\)应当记的是最短用时,因为用时越短,在的位置也越高,然后对后续转移也越有利,至于有可能要强制往下移,那只要在该移的时候向下移动就行,用时短一定不必用时长劣

还有一种情况,是移到下一棵树会移到\(<0\)的高度,这时候我们就要往上移,移到能飞到下一棵树的位置,然后往下一棵树移.移到下一棵树后高度会变为0,这时候对于任何移动,我们都是先往上移\(t_i(t_i\le H_x)\),然后飞到下一棵树,所以用时都是\(2t_i\).那么上面两种情况就可以分别用\(dij\)来转移了.注意如果在第一种移动过程中移到了\(n\),那么就可以更新答案了,所以最终答案为\(min(\)这种情况的贡献\(,dis_n+H_n)\)

#include<bits/stdc++.h>
#define LL long long
#define uLL unsigned long long
#define db long double

using namespace std;
const int N=1e5+10;
int rd()
{
    int x=0,w=1;char ch=0;
    while(ch<'0'||ch>'9'){if(ch=='-') w=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+(ch^48);ch=getchar();}
    return x*w;
};
int n,m,X,h[N],e[N*3][3];
struct node
{
    LL x,d;
    bool operator < (const node &bb) const {return d>bb.d;}
};
priority_queue<node> q,qq;
int to[N<<4],nt[N<<4],w[N<<4],hd[N],tot=1;
void add(int x,int y,int z)
{
    ++tot,to[tot]=y,nt[tot]=hd[x],w[tot]=z,hd[x]=tot;
}
LL di[N],ans=1ll<<50;

int main()
{
    n=rd(),m=rd(),X=rd();
    for(int i=1;i<=n;++i) h[i]=rd();
    for(int i=1;i<=m;++i)
        for(int j=0;j<=2;++j)
            e[i][j]=rd();
    for(int i=1;i<=m;++i)
    {
        int x=e[i][0],y=e[i][1],z=e[i][2];
        add(x,y,z),add(y,x,z);
    }
    memset(di,0x3f3f3f,sizeof(LL)*(n+1));
    qq.push((node){1,di[1]=0});
    LL he=X;
    while(!qq.empty())
    {
        int x=qq.top().x;
        LL d=qq.top().d;
        qq.pop();
        if(d>di[x]) continue;
        if(x==n) ans=min(ans,di[x]+h[n]-(he-di[x]));
        for(int i=hd[x];i;i=nt[i])
        {
            int y=to[i];
            if(he-di[x]-w[i]<=0&&w[i]<=h[x]) q.push((node){y,di[x]+w[i]-(he-di[x]-w[i])});
            else
            {
                LL dt=max(0ll,he-di[x]-w[i]-h[y]);
                if(di[y]>di[x]+w[i]+dt&&di[x]+w[i]+dt<=he) qq.push((node){y,di[y]=di[x]+w[i]+dt});
            }
        }
    }
    memset(hd,0,sizeof(int)*(n+1)),tot=1;
    for(int i=1;i<=m;++i)
    {
        int x=e[i][0],y=e[i][1],z=e[i][2];
        if(h[x]>=z) add(x,y,z<<1);
        if(h[y]>=z) add(y,x,z<<1);
    }
    memset(di,0x3f3f3f,sizeof(LL)*(n+1));
    while(!q.empty())
    {
        int x=q.top().x;
        LL d=q.top().d;
        q.pop();
        if(d>di[x]) continue;
        if(d<di[x]) di[x]=d;
        for(int i=hd[x];i;i=nt[i])
        {
            int y=to[i];
            if(di[y]>di[x]+w[i])
                q.push((node){y,di[y]=di[x]+w[i]});
        }
    }
    ans=min(ans,di[n]+h[n]);
    printf("%lld\n",ans<(1ll<<50)?ans:-1);
    return 0;
}

原文地址:https://www.cnblogs.com/smyjr/p/11618543.html

时间: 2024-10-30 04:29:04

loj 2759「JOI 2014 Final」飞天鼠的相关文章

loj 3014「JOI 2019 Final」独特的城市

loj 我本来是直接口胡了一个意思一样的做法的,但是因为觉得有点假+实现要用并查集(?)就卡了好一会儿... 对于一个点\(x\)来说,独特的点一定在它的最长链上,如果有独特的点不在最长链上,那么最长链上一定有和他到\(x\)距离相同的点,矛盾 然后对于一个点,最长链端点一定可以是直径的两端点之一,所以如果我们分别以树的直径的两端点为根进行dfs,那么一个点在其中一次dfs中,独特的点都会在到根的路径上,所以我们用栈维护到根的点,然后不同颜色数开桶来维护,每次压栈或弹栈时改变桶内元素个数,然后根

【2018.10.1】「JOI 2014 Final」年轮蛋糕

题面 一看到求“最小值的最大值”这种问题,就能想到二分了. 二分答案,然后判然后把环展开成2倍长度的链,先钦定一个起点, 原文地址:https://www.cnblogs.com/scx2015noip-as-php/p/9734669.html

loj#2334 「JOI 2017 Final」JOIOI 王国

分析 二分答案 判断左上角是否满足 为了覆盖所有范围 我们依次把右下角,左上角,右上角移动到左上角 代码 #include<bits/stdc++.h> using namespace std; int a[2010][2010],n,m,Ans=1e9+7,mx,mn=1e9+7; inline bool ck(int x){ int i,j,k,l=mn+x,r=mx-x,lim=m; for(i=1;i<=n;i++){ for(j=1;j<=lim;j++) if(a[i]

Libre OJ P2332「JOI 2017 Final」焚风现象【差分思想】By cellur925

题目传送门 这道题开始看起来会很晕...\(qwq\).首先我们要明确题目中的海拔&&温度.温度是受海拔影响的,每次改变的是海拔,我们求的是温度. 我们开始读入的时候便可以处理出开始\(N\)位置的温度以及各个位置的海拔差.每次读入影响的是一段区间,区间内的相对海拔是不变的因此温度也不会变.只有区间的边界可能受到影响.因此我们只要处理边界就行了:这便是差分的思想. 比如有\([l,r]\)区间需要处理,那么我们把\(l\)位置的原答案减去,把\(l\)位置的海拔改变,并加上新答案.再对\(

[LOJ#2325]「清华集训 2017」小Y和恐怖的奴隶主

[LOJ#2325]「清华集训 2017」小Y和恐怖的奴隶主 试题描述 "A fight? Count me in!" 要打架了,算我一个. "Everyone, get in here!" 所有人,都过来! 小Y是一个喜欢玩游戏的OIer.一天,她正在玩一款游戏,要打一个Boss. 虽然这个Boss有 \(10^{100}\) 点生命值,但它只带了一个随从--一个只有 \(m\) 点生命值的"恐怖的奴隶主". 这个"恐怖的奴隶主&qu

[LOJ#2328]「清华集训 2017」避难所

[LOJ#2328]「清华集训 2017」避难所 试题描述 "B君啊,你当年的伙伴都不在北京了,为什么你还在北京呢?" "大概是因为出了一些事故吧,否则这道题就不叫避难所了." "唔,那你之后会去哪呢?" "去一个没有冬天的地方." 对于一个正整数 \(n\),我们定义他在 \(b\) 进制下,各个位上的数的乘积为 \(p = F(n, b)\). 比如 \(F(3338, 10) = 216\). 考虑这样一个问题,已知 \

[LOJ#2327]「清华集训 2017」福若格斯

[LOJ#2327]「清华集训 2017」福若格斯 试题描述 小d是4xx9小游戏高手. 有一天,小d发现了一个很经典的小游戏:跳青蛙. 游戏在一个 \(5\) 个格子的棋盘上进行.在游戏的一开始,最左边的两个格子上各有一个向右的青蛙,最右边的两个格子上各有一个向左的青蛙. 每次移动可以选取一个青蛙,向这只青蛙的前方移动一格到空格子中或跳过前方的一个不同朝向的青蛙并移动到空格子中. 为了使你更好地理解这个游戏,我们下发了一个游戏demo作为参考(注意:这个demo中的棋盘大小和题目中并不相同).

[LOJ#2331]「清华集训 2017」某位歌姬的故事

[LOJ#2331]「清华集训 2017」某位歌姬的故事 试题描述 IA是一名会唱歌的女孩子. IOI2018就要来了,IA决定给参赛选手们写一首歌,以表达美好的祝愿.这首歌一共有 \(n\) 个音符,第iii个音符的音高为 \(h_i\).IA的音域是 \(A\),她只能唱出 \(1\sim A\) 中的正整数音高.因此 \(1\le h_i\le A\). 在写歌之前,IA需要确定下这首歌的结构,于是她写下了 \(Q\) 条限制,其中第 \(i\) 条为:编号在 \(l_i\) 到 \(r_

loj #6091. 「Codeforces Round #418」幻想特快

#6091. 「Codeforces Round #418」幻想特快 1 #include<algorithm> 2 #include<iostream> 3 #include<cstring> 4 #include<cstdio> 5 #include<queue> 6 using namespace std; 7 #define maxn 10000 8 int n,a[maxn],b[maxn],p[maxn],nxt1,nxt2,tot,