说一个代码简单的做法。本题解数形结合,容易理解。
题解
首先,题目所求就是上述不等式对哪个 $i$ 成立。
发现式中只有第 $2$ 项是随着 $i$ 而变化的。而第 $1$ 项虽然不变,但是较为复杂,与第 $2$ 项相减的结果不好快速计算,因此不妨将第 $2$ 项移项到右边。
是不是突然发现这下不等式两边都比较简单了?如果看不出来,我们不妨把式子左右的图叠在一起看:
下文中称 $(x_i,p_i+m)$ 点为蓝色山顶(图中标出)
我们考虑右边能把左边盖住的条件:
- 对于左边高度小于等于 $m$ 的点,显然不用考虑
- 对于左边高度大于 $m$ 的某一点,蓝色山顶必须位于下图红色阴影区域内:
显然,只要左边的所有极值点都满足上述条件,那么左边的所有点都满足上述条件。到了这里,做法就显而易见了。我们只需要枚举所有左侧的极值点,然后把他们的红色阴影区域求交,再判断某个 $i$ 对应的蓝色山顶是否在这个区域内即可。
至于维护红色阴影的交,可以把这个区域看成两个斜率分别为 $\pm1$ 的一次函数取 max,因此求交的时候只需要对应截距取 max 即可。
剩下的就是一些简单的技术问题,比如离散化,差分,不再赘述。
复杂度 $O(n\log n)$,瓶颈在离散化。
代码
#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
using ll = long long;
void umx(ll &x, const ll &y) { x = max(x, y); }
const char nl = '\n';
const ll INF = 1e18;
const ll MXN = 1e6 + 5;
ll n, m;
map<ll, ll> delt;
ll x[MXN], p[MXN];
int main() {
ios::sync_with_stdio(0);
cin.tie(0);
int t;
cin >> t;
while (t--) {
delt.clear();
cin >> n >> m;
for (ll i = 1; i <= n; i++) {
cin >> x[i] >> p[i];
delt[x[i] - p[i] + 1]++;
delt[x[i] + 1] -= 2;
delt[x[i] + p[i] + 1]++;
}
ll b = 0, k = 0, lastx = -INF;
ll b_1 = -INF, b1 = -INF;
for (auto it : delt) {
b += k * (it.fi - lastx);
k += it.se;
if (b > m) {
umx(b1, b - it.fi + 1);
umx(b_1, b + it.fi - 1);
}
lastx = it.fi;
}
for (ll i = 1; i <= n; i++) cout << ((p[i] + m - x[i] >= b1) && (p[i] + m + x[i] >= b_1));
cout << nl;
}
return 0;
}
作者:@ethan-enhe
本文为作者原创,转载请注明出处:本文链接
阅读量: