专栏文章
消失吧,群青
P12922题解参与者 1已保存评论 0
文章操作
快速查看文章及其快照的属性,并进行相关操作。
- 当前评论
- 0 条
- 当前快照
- 1 份
- 快照标识符
- @mindwfkg
- 此快照首次捕获于
- 2025/12/02 00:49 3 个月前
- 此快照最后确认于
- 2025/12/02 00:49 3 个月前
发现时刻和位置可以非整数很难处理,由于精度要求不高,我们把时间和长度尺度全部乘以 ,只考虑在整数情况,最后把答案除以 ,这样数值上能近似求出在任意时刻都可以转弯的答案,这里 取 足够了。
二分答案 。把流星按照时间排序。发现每个时刻能到达的点集是一些区间的并,我们用一个
CPPset 维护这些区间,时间经过 会让这些区间的左右端点各延伸 单位长度,并合并相邻有交的区间,在坐标 上落下流星等于该时刻不能位于 ,需要点集内在该区间内的点删除。删除操作就类似 ODT 直接可以做完,每次暴力延伸区间时间复杂度不可行,所以我们给每个区间再带上一个标记表示上次被更新的时刻,删除操作只关心和 有交的区间,我们只更新左右端点前后的区间即可。然后做完了,复杂度 。#include <bits/stdc++.h>
#define LL long long
#define ull unsigned long long
#define uint unsigned int
using namespace std;
const int N = 2e5 + 10;
int n; pair<LL, LL> A[N];
struct Info { LL l, r, t; };
const bool operator<(const Info a, const Info b) {
return (a.l != b.l ? a.l < b.l : (a.r != b.r ? a.r < b.r : a.t < b.t));
}
set<Info> s;
void upd(Info x, LL t) {
auto it = s.find(x);
if (it == s.end()) return ;
auto [l, r, t0] = *it; r += t - t0; l -= t - t0; it = s.erase(it);
while (it != s.end() && it->l - r <= t - it->t)
r = max(r, it->r + t - it->t), it = s.erase(it);
while (it != s.begin()) {
it = prev(it); if (l - it->r > t - it->t) break;
l = min(l, it->l - t + it->t); it = s.erase(it);
}
s.insert({l, r, t});
}
void upd(LL x, LL t) {
auto it = s.lower_bound({x + 1, 0, 0}); upd(*it, t);
it = s.lower_bound({x + 1, 0, 0}); if (it != s.begin()) upd(*prev(it), t);
}
bool check(LL d) {
if (d == 0) return true;
s.clear(); s.insert({0, 0, 0});
for (int i = 1; i <= n; i ++) {
if (s.empty()) return false;
auto [cur, x] = A[i];
LL l = x - d + 1, r = x + d - 1;
upd(r, cur); upd(l, cur);
auto it = s.lower_bound({l + 1, 0, 0});
if (it != s.begin() && prev(it)->r >= l) {
auto [a, b, c] = *prev(it); s.erase(prev(it));
if (a < l) s.insert({a, l - 1, c});
if (b > r) { s.insert({r + 1, b, c}); continue; }
}
while (it != s.end() && it->l <= r) {
auto [a, b, c] = *it; it = s.erase(it);
if (b > r) { s.insert({r + 1, b, c}); break; }
}
}
return !s.empty();
}
int main() {
freopen(".in", "r", stdin); freopen(".out", "w", stdout);
ios::sync_with_stdio(false); cin.tie(0), cout.tie(0);
cin >> n; const ull K = 1000;
for (int i = 1, t, x; i <= n; i ++)
cin >> t >> x, A[i] = {1ll * t * K, 1ll * x * K};
sort(A + 1, A + 1 + n);
LL L = 0, R = 1e13;
while (L + 1 < R) {
LL mid = (L + R) >> 1;
if (check(mid)) L = mid;
else R = mid;
}
cout << fixed << setprecision(1) << (double)L / K << "\n";
return 0;
}
相关推荐
评论
共 0 条评论,欢迎与作者交流。
正在加载评论...