洛谷P1258 小车问题 题解

题目描述

甲、乙两人同时从 A 地出发要尽快同时赶到 B 地。出发时 A 地有一辆小车,可是这辆小车除了驾驶员外只能带一人。已知甲、乙两人的步行速度一样,且小于车的速度。问:怎样利用小车才能使两人尽快同时到达。

输入格式

仅一行,三个实数,分别表示 AB 两地的距离 𝑠,人的步行速度 𝑎,车的速度 𝑏。

输出格式

两人同时到达 B 地需要的最短时间,保留 6 位小数。

输入输出样例

输入 #1

120 5 25

输出 #1

9.600000

说明/提示

数据规模与约定

对于 100% 的数据,保证 0≤𝑠,𝑎,𝑏≤10^9

题解:

首先可以贪心地得到,一定是车先载一人走一段距离 x 放下,再回头载另一人到B地,此时两人同时到达B地,且时间最短。因为要让两人坐车的时间尽量长,不能让时间多花在车的往返上。

那么最短时间就等于 x / a + (s - x) / b,于是关键在于求 x。官方题解给了这样一个式子:(s - x) / a = 2 * (2 * x / (a + b) - x / b) + (s - x) / b,其他的还好说,这个 2 * x / (a + b) 是哪来的?下面给出我的解释。

先思考这样一个问题:一人一车分别以速度 a,b(a < b)从 0 坐标出发,在 x 轴上 0 到 x 之间来回走,走到尽头就回头继续走,那么如何求他们的第一次相遇位置 y ?

一个办法是,车到 x 处时,人已走距离 ax/b ,车从x到y的时间等于人从 ax/b 到 y 的时间,有 (x - y) / b = (y - ax/b) / a,可解出 y = 2ax / (a + b)。但有个更快的办法:假设车是从 2x 坐标出发,向 x 轴负方向走,对于求 y 而言与原情况完全等效。于是相遇时间为 2x / (a + b),y = 2ax / (a + b)。

回到原问题,根据上面得到的结论,从车放下人返回到车与另一个人相遇花了 2x / (a + b) 的时间。从车放下人算起,两人要同时到达B地,有 (s - x) / a = 2 * (2 * x / (a + b) - x / b) + (s - x) / b,化简(别直接去分母,先因式分解计算量很小)得 x = (a + b) * s / (3 * a + b) ,代入 x / a + (s - x) / b 就求得了最短时间。附ac代码(注意保留6位小数):

#include<bits/stdc++.h>
#define nn '\n'
typedef long long ll;
using namespace std;

// const int maxn = 2e5 + 5;
// bool vis[maxn];

void solve() {
    double s, a, b;
    cin >> s >> a >> b;
    double x = (a + b) * s / (3 * a + b);
    cout << fixed << setprecision(6) << x / b + (s - x) / a << nn; // 保留6位小数
}
int main() {
    ios::sync_with_stdio(false);
    cin.tie(0);
    cout.tie(0);
    solve();
    return 0;
}

  • 5
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值