题意
给出一个长度为的序列,求对于所有,每次从序列中选出个互不相同的数,最多能取多少次。
题解
我们首先把数组转化为相同的数的出现次数的序列,如序列就转化为。把这个得到的序列计作。
然后二分答案,假设当前二分到,能取次的条件是:
因为同一个数最多取到次,而取到的总数为。
发现当增大时答案减小,所以可以把二分去掉,写一个指针动就行了。这样的时间复杂度是的,因为要。
CODE
#include <bits/stdc++.h>
using namespace std;
inline void rd(int &x) {
char ch; for(;!isdigit(ch=getchar()););
for(x=ch-'0';isdigit(ch=getchar());)x=x*10+ch-'0';
}
typedef long long LL;
const int MAXN = 300005;
int N, n, a[MAXN], cnt[MAXN];
LL sum[MAXN];
int ans[MAXN];
inline bool chk(int k, int x) {
int pos = upper_bound(a + 1, a + n + 1, x) - a;
return sum[pos-1] + 1ll*(n-pos+1)*x >= 1ll*k*x;
}
int main() {
rd(N);
for(int i = 1, x; i <= N; ++i) rd(x), ++cnt[x];
for(int i = 1; i <= 300000; ++i) if(cnt[i]) a[++n] = cnt[i];
sort(a + 1, a + n + 1);
for(int i = 1; i <= n; ++i) sum[i] = sum[i-1] + a[i];
int now = 0;
for(int k = n; k >= 1; --k) {
while(now < N && chk(k, now+1)) ++now;
ans[k] = now;
}
for(int i = 1; i <= N; ++i) printf("%d
", ans[i]);
}
然而看了别人的博客后发现,可以。
上代码:
存的就是上面的不等式的左边部分。
#include <bits/stdc++.h>
using namespace std;
inline void rd(int &x) {
char ch; for(;!isdigit(ch=getchar()););
for(x=ch-'0';isdigit(ch=getchar());)x=x*10+ch-'0';
}
typedef long long LL;
const int MAXN = 300005;
int n, cnt[MAXN];
LL sum[MAXN];
int ans[MAXN];
inline bool chk(int k, int x) { return sum[x] >= 1ll*k*x; }
int main() {
rd(n);
for(int i = 1, x; i <= n; ++i) rd(x), ++cnt[x], ++sum[cnt[x]];
for(int i = 1; i <= n; ++i) sum[i] += sum[i-1];
int now = 0;
for(int k = n; k >= 1; --k) {
while(now < n && chk(k, now+1)) ++now;
ans[k] = now;
}
for(int i = 1; i <= n; ++i) printf("%d
", ans[i]);
}
正确性读者自证不难