上海计算机学会 2024年3月月赛 丙组T5 限速(思维、数学)

文章讨论了如何计算在一条路上开车,根据给定的限速点和初始速度,如何在满足限速要求下最大化行驶过程中的最大速度。通过动态规划方法,从后往前计算每个限速点的可行速度并更新最大速度。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

第五题:T5限速

标签:思维、数学
题意:开车在一条路上,从坐标 0 0 0开始,车的起始速度是 1 1 1,车每开出一单位的距离,可以选择速度不变、或将速度增加一个单位、或减少一个单位,车的速度不能低于 1 1 1
车子将通过 n n n个限速点,第 i i i个测速点的坐标为 x i x_i xi,限速为 L i L_i Li,当通过最后一个限速点,结束旅程,求整个路程过程中,车子能够达到的最大速度是多少?
题解:需要先从后往前维护一下,降速情况下,通过每个限速点的最大速度,实际能否到达题目限制的 L i L_i Li,还会受到两个位置坐标距离的影响。
比如两个坐标位置距离为 10 10 10,前一个位置的限速为 30 30 30,后一个位置的限速为 5 5 5,那么开到前一个位置就不能是 30 30 30,而应该是 15 15 15,这样才能保证能到后一个位置能降速到 5 5 5
预处理好之后,从前往后遍历,维护一下通过第 i i i个限速点的最大可行速度, c [ i ] = m i n ( c [ i − 1 ] + x [ i ] − x [ i − 1 ] , L [ i ] ) c[i]=min(c[i-1]+x[i]-x[i-1],L[i]) c[i]=min(c[i1]+x[i]x[i1],L[i])

以下为数学推导:
a a a x i − 1 x_{i-1} xi1 x i x_i xi之间加速路段, b b b x i − 1 x_{i-1} xi1 x i x_i xi之间减速路段。
c i − 1 c_{i-1} ci1为第 i − 1 i-1 i1个点的可行速度, c i c_i ci为第 i i i个点的可行速度。
可知 a + b = x i − x i − 1 a+b=x_i-x_{i-1} a+b=xixi1 c i − 1 + a = c i + b c_{i-1}+a=c_i+b ci1+a=ci+b
c i − 1 + a = c i + x i − x i − 1 − a c_{i-1}+a=c_i+x_i-x_{i-1}-a ci1+a=ci+xixi1a
a a a移到左边: 2 a = c i − c i − 1 + x i − x i − 1 2a=c_i-c_{i-1}+x_i-x_{i-1} 2a=cici1+xixi1=> a = ( c i − c i − 1 + x i − x i − 1 ) / 2 a=(c_i-c_{i-1}+x_i-x_{i-1})/2 a=(cici1+xixi1)/2
最大速度为 c i − 1 + a c_{i-1}+a ci1+a,所以可以求出在这个路程中的最大速度为 ( c [ i ] + c [ i − 1 ] + x [ i ] − x [ i − 1 ] ) / 2 (c[i]+c[i-1]+x[i]-x[i-1])/2 (c[i]+c[i1]+x[i]x[i1])/2
代码

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

typedef long long ll;
const ll N = 3e5 + 10;

ll n, x[N], L[N], c[N];

int main() {
    cin >> n;
    for (int i = 1; i <= n; i++) {
        cin >> x[i] >> L[i];
    }
    for (int i = n; i >= 2; i--) {
        if (L[i] < L[i - 1]) {
            ll d = (x[i] - x[i - 1]);
            // 通过每个限速点的最大速度
            L[i - 1] = min(L[i - 1], L[i] + d);
        }
    }

    c[0] = 1;
    ll ans = 1;
    for (int i = 1; i <= n; i++) {
        ll d = (x[i] - x[i - 1]);
        // c[i]: 通过第i个限速点的最大可行速度
        c[i] = min(c[i - 1] + d, L[i]);
        ans = max(ans, (c[i] + c[i - 1] + d) / 2);
    }

    cout << ans;
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值