砍树
伐木工人 Mirko 需要砍?MM?米长的木材。对 Mirko 来说这是很简单的工作,因为他有一个漂亮的新伐木机,可以如野火一般砍伐森林。不过,Mirko 只被允许砍伐一排树。 Mirko 的伐木机工作流程如下:Mirko 设置一个高度参数?HH(米),伐木机升起一个巨大的锯片到高度?HH,并锯掉所有树比?HH?高的部分(当然,树木不高于?HH?米的部分保持不变)。Mirko 就得到树木被锯下的部分。例如,如果一排树的高度分别为?20,15,1020,15,10?和?1717,Mirko 把锯片升到?1515?米的高度,切割后树木剩下的高度将是?15,15,1015,15,10?和?1515,而 Mirko 将从第?11?棵树得到?55?米,从第?44?棵树得到?22?米,共得到?77?米木材。 Mirko 非常关注生态保护,所以他不会砍掉过多的木材。这也是他尽可能高地设定伐木机锯片的原因。请帮助 Mirko 找到伐木机锯片的最大的整数高度?HH,使得他能得到的木材至少为?MM?米。换句话说,如果再升高?11?米,他将得不到?MM?米木材。
第?11?行?22?个整数?NN?和?MM,NN?表示树木的数量,MM?表示需要的木材总长度。 第?22?行?NN?个整数表示每棵树的高度。
11?个整数,表示锯片的最高高度。
4 7
20 15 10 17
15
5 20
4 42 40 26 46
36
对于?100% 的测试数据,1≤N≤10^6,1≤M≤2×10^9,树的高度?<10^9,所有树的高度总和?>M。
上次我们讲了二分查找,这回我们来讲二分查找的运用
根据标题,这道题是要用到二分查找的,但二分查找怎么运用进去呢?
我们想一想,我们能想到的最简单粗暴的方法是什么?
没错,就是用一个循环一个一个的试,但一个一个的试很慢,如果我们用二分查找的话就会快很多
具体的运用是这样的:
首先写一个函数f,f(n)表示将高度设为n能砍到多少木头
那我们查找的范围,也就是答案的范围是什么?当然是1~mm(mm=最高的数的高度)
然后,我们就可以开始二分查找了,先找1~mm最中间的数,然后一直找下去(如果最中间的数砍不到m米的木头,就往低处,也就是1~最中间的数,否则就往高出,最中间的数~mm(就算砍到了m米的木头也不能停,继续往高处找!前进!不择手段地前进!))
然后输出right就好了(你不知道right是什么?赶紧看代码或教程!)
#include<cstdio>
long long n,m,high[1000000+10];
long long tmp,left,right;
int main(){
scanf("%lld%lld",&n,&m);
for(long long i=1;i<=n;i++){
scanf("%lld",&high[i]);
if(high[i]>right) right=high[i];//找最高的树的高度
}
while(left<=right){
int mid=(left+right)/2;
tmp=0;
for(int i=1;i<=n;i++)
if(high[i]>mid) tmp+=high[i]-mid;//这里我就没有写函数,直接用了循环
if(tmp<m) right=mid-1;//砍不到数就往低处找
else left=mid+1;
}
printf("%lld",right);//输出
return 0;
}