解题思路:

来x去x次,等价于来2*x次(即去的x次每次反向走,变成来的x次)。

一个青蛙跳2*x次,等价于2*x个青蛙同时跳一次。

贪心算法,一次跳跃尽可能跳最远的那个石头。
二分法,设跳跃能力y,每一段连续长度y内石头总数都要超过2*x,才能过河;若有一个区间y内石头总数小于2*x,该区间前一个区间的2*x个青蛙就无法全部通过这个区间。

注意事项:
数组类型,sum可能会超过int,故用long long
参考代码:

#include <iostream>
#include <cstdio>
#include <cstring>//strlen、strcmp 
#include <cmath>
#include <cstdlib>//malloc
#include <algorithm>
using namespace std;

#define maxn 100010
int n,x;//n-1个石头,2*x次通过 
int h[maxn];
long long sum[maxn];//每块石头高度、高度的前缀和数组 
int y;//弹跳能力 
bool above(int a){//每段区间长度为a时,判断其内总石头数是否>=2*x
//注意是任意一段连续的长度为a的区间都要满足才为真,并不是分块 
	for(int i = 1;i <= n-a;++i){
		int r = i+a-1;
		if(sum[r]-sum[i-1]<2*x){
			return false;
		}
	}
	return true;
}
int main() {
	scanf("%d%d",&n,&x);
	for(int i = 1;i < n;++i){
		scanf("%d",&h[i]);
		sum[i] = sum[i-1] + (long long)h[i];
	}
	y = n;
	int l=1,r=n,mid;
	while(l < r){//二分法,类似lower_bound 
		mid=l+(r-l)/2;
		if(above(mid)){//若弹跳能力为mid时可以,则y最多为mid,查找左区间 
			r = mid; 
		}else{//若mid时不行,则y必大于mid,查找右区间 
			l = mid+1;
		}
	}//最后退出为l==r 
	y = l;
	printf("%d\n",y); 
	
		
	return 0;
}


点赞(1)
 

0.0分

2 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论