CF506C Mr. Kitayuta vs. Bamboos

CF506C

n n n个竹子,一开始的高度为 h i h_i hi,每天末尾生长 a i a_i ai

m m m天,每天中间你可以选择 k k k棵竹子(可以重复选),使得它们的高度减 p p p,如果减到负数就变为 0 0 0,但是这个竹子没有消失。

求第 m m m天末最高的竹子最矮多少。

n ≤ 1 0 5 n\le 10^5 n105


巧妙的转化。

显然二分答案,转化成判定性问题。我们需要找到一种方案满足最终所有的竹子的高度都小于等于 m i d mid mid

如何判定?这里用了个奇妙的转化:如果将时间倒流,就可以视为,每过一天,竹子会缩短 a i a_i ai,砍竹子视为将竹子拉高 p p p,中间不可以出现负数,最终所有的竹子的高度都为 h i h_i hi

既然二分了,那就放缩一下:一开始所有竹子的高度为 m i d mid mid,要求最终的高度大于等于 h i h_i hi

这样转化的好处时:砍竹子的操作无限制,每次一定砍 p p p

于是按照时间从后往前枚举,计算每个竹子缩多少次就会变为负数,如果这个次数小于等于 m m m,就丢进小根堆里。每次取堆顶进行操作。最后再用剩下的操作次数将竹子的高度调到 h i h_i hi上即可。


using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 100010
#define ll long long
int n,m,k;
ll p;
ll h[N],a[N];
ll s[N];
int q[N],nq;
bool cmpq(int x,int y){return s[x]/a[x]>s[y]/a[y];}
bool ok(ll lim){
	nq=0;
	for (int i=1;i<=n;++i){
		s[i]=lim;
		if (s[i]/a[i]+1<=m)
			q[nq++]=i;
	}
	make_heap(q,q+nq,cmpq);
	ll rem=k*m;
	for (int i=1;i<=m && nq;++i){
		for (int j=1;j<=k && nq;++j){
			rem--;
			int x=q[0];
			pop_heap(q,q+nq--,cmpq);
			if (s[x]-i*a[x]<0) return 0;
			s[x]+=p;
			if (s[x]/a[x]+1<=m){
				q[nq++]=x;
				push_heap(q,q+nq,cmpq);
			}
		}
	}
	if (nq) return 0;
	for (int i=1;i<=n;++i){
		s[i]-=a[i]*m;
		if (s[i]>=h[i]) continue;
		ll tmp=(h[i]-s[i]-1)/p+1;
		rem-=tmp;
		if (rem<0) return 0;
	}
	return 1;
}
int main(){
	scanf("%d%d%d%lld",&n,&m,&k,&p);
	for (int i=1;i<=n;++i)
		scanf("%lld%lld",&h[i],&a[i]);
	ll l=0,r=1e18;
	while (l<r){
		ll mid=l+r>>1;
		if (ok(mid))
			r=mid;
		else
			l=mid+1;
	}
	ok(0);
	printf("%lld\n",r);
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值