F - Distinct Numbers 每次取k个不同的 整零计算

这种题目的特征。就是每次拿k个 然后每个都要不一样🔥。。
问最多能拿多少次。。

这个数据预处理还是挺方便计算的。。
主要是理解 b[x] + x * (c[n] - c[x]) >= x * y;
b c数组 都是一个 递增的梯形 。。
他这个验证的意思是。。从x位置截断 前面按照零的算 后面按照整的算🦋
整的就是算x的高度。零的就拼起来算。。----这是这种题的计算特点。
然后这样算起来 是否大于x*y所形成的矩形。。这可以参考下面那篇题解。

b数组实际上是 高度sum 。。。是为了取零的。。
c数组是 值x的cnt 的cnt 前缀。—没错就是两个cnt嵌套含义。

#include <bits/stdc++.h>
using namespace std;
#define int long long
#define ll __int128_t
#define ar array<int, 2>
#define arr array<int, 3>
int  n, m, k, inf = 1LL << 61, mod = 998244353;// 1e9+7;
const int N = 5e5 + 50;
int a[N], b[N], c[N];
void solve() {
	cin >> n;
	for (int i = 1; i <= n; ++i) {
		int x; cin >> x;
		a[x]++;
	}
	for (int i = 1; i <= n; ++i) {
		b[a[i]] += a[i];
		c[a[i]]++;
	}
	for (int i = 1; i <= n; ++i) {
		c[i] += c[i - 1];
		b[i] += b[i - 1];
	}

	auto ok = [&](int x, int y) {
		return b[x] + x * (c[n] - c[x]) >= x * y;
	};

	for (int i = 1; i <= n; ++i) {
		int l = 0, r = n;
		while (l < r) {
			int mid = (l + r + 1) >> 1;
			if (ok(mid, i))
				l = mid;
			else
				r = mid - 1;
		}
		cout << l << '\n';
	}
};
// https://blog.csdn.net/qq_39973668/article/details/102925705
// 就是 整 零。。那个模型。。只不过这里。。更抽象了
//这题主要跑不了nn。。。
// 主要是怎么快速的算。。。。。
//https://www.cnblogs.com/Accepting/p/11719466.html






signed main() {
	ios::sync_with_stdio(false);
	cin.tie(0);
	cout << fixed << setprecision(15);
#ifdef DEBUG
	freopen("../1.in", "r", stdin);
#endif
	//init_f();
	//init();
	//expr();
	// int T; cin >> T; while(T--)
	solve();
	return 0;
}



  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值