1 条题解
-
3
做法
我们先把所有的按照进行排序,可以发现答案肯定是小于等于、大于等于的。所以我们先设。接下来我们考虑剩下的序列如何对进行更新。由题,对于某个,如果说的话,那么就可以对答案进行更新。化简得。我们设为此时的所有区间中小于的区间数。易得是非严格单调函数。所以此不等式可以转换成,即,。此时的值循环++,直到不满足条件再枚举下一个 接下来我们来讨论如何快速求出的值,可以使用双指针解决这个问题。对于每个右端点,存在左端点使得 ,且随非严格单调递增。所以我们可以使用一个桶来维护每个数在间出现的次数。再用一个变量来记录间值不等于的个数。当时代表在中包含中的所有数,此时左端点右移直至。如图
时间复杂度
每次的时间复杂度为,因为,所以最多加次,内层while复杂度为,外层for循环的时间复杂度也为。所以综合的时间复杂度为
#include <bits/stdc++.h> using namespace std; const int N = 1e4 + 10; int n, a[N]; int ton[N], cnt; struct node { int k, w, m; unsigned long long seed; } q[N]; int k, w, m; bool cmp(node x, node y) { if (x.w != y.w) return x.w > y.w; return x.m > y.m; } unsigned long long _rand(unsigned long long &seed) { seed ^= seed << 13; seed ^= seed >> 7; seed ^= seed << 17; return seed; } inline int check(int mid) { memset(ton, 0, sizeof ton); int res = 0; cnt = 0; int l = 1, r = 1; for (; r <= n; r++) { if (ton[a[r]] == 0) { if (a[r] < mid) { cnt++; ton[a[r]]++; while (l <= r && cnt == mid) { ton[a[l]]--; if (ton[a[l]] == 0) cnt--; l++; } } } else ton[a[r]]++; res += r - l + 1; } return res; } int main() { scanf("%d", &n); for (int i = 1; i <= n; i++) scanf("%d%d%d%llu", &q[i].k, &q[i].w, &q[i].m, &q[i].seed); sort(q + 1, q + 1 + n, cmp); int ans = q[1].w; for (int i = 1; i <= n; i++) { k = q[i].k, w = q[i].w, m = q[i].m; unsigned long long se = q[i].seed; for (int j = 1; j <= n; j++) { a[j] = _rand(se) % q[i].m; } while (check(ans - w + 1) <= k) ans++; } printf("%d\n", ans); return 0; }
信息
- ID
- 154
- 时间
- 1500ms
- 内存
- 512MiB
- 难度
- 9
- 标签
- 递交数
- 223
- 已通过
- 16
- 上传者