蓝桥杯复习之二分法与前缀和

题目:最佳牛围栏

题目链接:https://www.acwing.com/problem/content/104/

题意:农夫约翰的农场由 N 块田地组成,每块地里都有一定数量的牛,其数量不会少于 1 头,也不会超过 2000 头。

约翰希望用围栏将一部分连续的田地围起来,并使得围起来的区域内每块地包含的牛的数量的平均值达到最大。

围起区域内至少需要包含 F 块地,其中 F 会在输入中给出。

在给定条件下,计算围起区域内每块地包含的牛的数量的平均值可能的最大值是多少?

思路:

找平均值/最小(大)值的最大(小)值,往二分答案方向想,二分答案的模板见代码。我们在每一次假设答案的计算过程中,将数组都减去mid方便计算(也就是只要存在满足条件同时大于0的字段就可以返回true了)

因为是找连续的一段,我们用前缀和优化计算,假设前缀和数组为pre,只要存在pre[j]-pre[i]>=0且j-i>=f就可以返回true了。

cpp 复制代码
#include<bits/stdc++.h>

using namespace std;

using ll = long long;
const ll N = 100005;
const ll mod = 1e9 + 7;
const int maxn = 2005;
int gcd(int a, int b) {
	return b ? gcd(b, a % b) : a;
}

int n, f;
int a[N];
double pre[N];

bool check(double m) {
	for (int i = 1; i <= n; i++) {
		pre[i] = pre[i - 1] + a[i] * 1.0 - m;
	}
	
	double minn = 0;

	for (int i = 0, j = f; j <= n; j++, i++) {
		minn = min(minn, pre[i]);
		if (pre[j] >= minn)
			return true;
	}
	return false;
}

void solve() {
	cin >> n >> f;

	for (int i = 1; i <= n; i++) {
		cin >> a[i];
	}

	double l = 1, r = 2002;
	while (r-l>=1e-5) {
		double mid = (l + r) / 2;
		if (check(mid)) {
			l = mid;
		}
		else
			r = mid;
	}
	cout << (int)(r * 1000) << '\n';

}
		

signed main() {

	ios::sync_with_stdio(false);
	cin.tie(0);
	std::cout.tie(0);

	int t = 1; //cin >> t;
	while (t--)
		solve();

	return 0;
}
相关推荐
memcpy013 小时前
LeetCode 2615. 等值距离和【相同元素分组+前缀和;考虑距离和的增量】中等
算法·leetcode·职场和发展
alphaTao14 小时前
LeetCode 每日一题 2026/4/20-2026/4/26
算法·leetcode·职场和发展
AC赳赳老秦17 小时前
OpenClaw实战案例:用Agent实现每日工作日报自动生成+发送
人工智能·python·职场和发展·eclipse·github·deepseek·openclaw
haina201917 小时前
海纳AI正式发布“面试Agent”——实现千岗千面与人机共管的智面新纪元
人工智能·面试·职场和发展
Morwit1 天前
【力扣hot100】 416. 分割等和子集
数据结构·c++·算法·leetcode·职场和发展
宵时待雨1 天前
优选算法专题3:二分查找
数据结构·c++·算法·leetcode·职场和发展
米粒11 天前
力扣算法刷题 Day 52
算法·leetcode·职场和发展
浅念-1 天前
LeetCode 模拟算法:用「还原过程」搞定编程题的入门钥匙
开发语言·c++·学习·算法·leetcode·职场和发展·模拟
阿Y加油吧1 天前
算法二刷复盘|旋转排序数组二分双杀(LeetCode 33 & 153)
算法·leetcode·职场和发展
skywalker_111 天前
力扣hot100(9-找到字符串中所有字母异位词;10-和为K的子数组)
算法·leetcode·职场和发展