给一个数组a,对于每一个元素,我们都会有一个初始值w = a[i] (0 <= i < n)只要w大于或等于数组中的某个元素,w = w + a[k]并且就会把这个数删除掉.
问: 对于每个元素,至多可以删除多少元素?
解题思路
由于需要最大化删除个数,我们可以先对数组升序排序,这样对于i(0 <= i < n) 就可以将0 ~ i-1的所有数删除掉,然后再考虑后面的.
就这样模拟下过程就完事儿了.
代码
(提交的一次TLE的代码)
#include<bits/stdc++.h>
#define int long long #define endl '\n' typedef std::pair<int, int> pii;
voidsolve(){ int n; std::cin >> n; pii a[n]; for (int i = 0; i < n; i++) std::cin >> a[i].first, a[i].second = i; std::sort(a, a + n); std::vector<int> s(n, 0ll); for (int i = 0; i < n; i++) { if (i == 0) s[i] = a[i].first; else s[i] = s[i - 1] + a[i].first; } std::vector<int> ans(n, 0ll); for (int i = 0; i < n; i++) { int t = s[i], j = i + 1; while (j < n && t >= a[j].first) t += a[j++].first; ans[i] = j - 1; } for (int i = 0; i < n; i++) a[i].first = ans[i]; std::sort(a, a + n, [&](pii aa, pii bb) { return aa.second < bb.second; }); for(int i = 0 ; i < n ; i ++){ std::cout << a[i].first << " "; } std::cout << endl; }
signedmain(){ std::ios::sync_with_stdio(false); std::cin.tie(nullptr), std::cout.tie(nullptr); int Lazy_boy_ = 1; std::cin >> Lazy_boy_; while (Lazy_boy_--) solve(); return0; }
(AC代码)
#include<bits/stdc++.h>
#define int long long #define endl '\n' [[maybe_unused]] constint INF = 1e9 + 50;
typedef std::pair<int, int> pii;
voidsolve(){ int n; std::cin >> n; pii a[n + 1]; for (int i = 0; i < n; i++) { std::cin >> a[i].first; a[i].second = i; } std::sort(a, a + n); std::vector<int> b(n, 0ll), c(n, 0ll), d(n, 0ll); for (int i = 0; i < n; i++) { b[i] = a[i].first; if (i == 0) c[i] = b[i]; else c[i] = c[i + 1] + b[i]; } for (int i = 0; i < n; i++) { int t = c[i]; int k = t; while (true) { int w = std::upper_bound(b.begin(), b.end(), t) - b.begin(); if (w == n) { d[a[i].second] = n - 1; break; } else { if (c[w - 1] == k) { d[a[i].second] = w - 1; break; } else { k = c[w - 1]; t = c[w - 1]; } } } }
for (int i = 0; i < n; i++) std::cout << d[i] << " "; std::cout << endl; }
signedmain(){ std::ios::sync_with_stdio(false); std::cin.tie(nullptr), std::cout.tie(nullptr); int Lazy_boy_ = 1; std::cin >> Lazy_boy_; while (Lazy_boy_--) solve(); return0; }
最后, 再是 k == 2时, 题目中明确告诉n^2 < 4e6,这就是让我们暴力循环,我们可以先将k == 1时的最小值记录下来,然后再暴力n^2跑一遍数组,再与之前的比较一下,最后输出答案.
代码
#include<bits/stdc++.h>
#define int long long #define endl '\n'
voidsolve(){ int n, k; std::cin >> n >> k; std::vector<int> a; for (int i = 0; i < n; i++) { int x; std::cin >> x; a.push_back(x); } if (k >= 3) { std::cout << 0 << endl; return; } std::sort(a.begin(), a.end()); auto calc = [&]() {//相当于处理了一次 std::sort(a.begin(), a.end()); int ans = a[0]; for (int i = 1; i < (int) a.size(); i++) { ans = std::min(ans, a[i] - a[i - 1]); } return ans; };
int ans = calc(); if (k == 2) { for (int i = 0; i < n; i++) { for (int j = i + 1; j < n; j++) { int w = a[j] - a[i]; int p = std::lower_bound(a.begin(), a.end(), w) - a.begin(); if (p != n - 1) ans = std::min(ans, a[p] - w); if (p != 0) ans = std::min(ans, w - a[p - 1]); } } } std::cout << ans << endl; }
signedmain(){ std::ios::sync_with_stdio(false); std::cin.tie(nullptr), std::cout.tie(nullptr); int Lazy_boy_ = 1; std::cin >> Lazy_boy_; while (Lazy_boy_--) solve(); return0; }