欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 健康 > 养生 > 每日c/c++题 备战蓝桥杯(洛谷P1873 EKO砍树问题详解)

每日c/c++题 备战蓝桥杯(洛谷P1873 EKO砍树问题详解)

2025/5/25 22:33:49 来源:https://blog.csdn.net/m0_74096349/article/details/148185247  浏览:    关键词:每日c/c++题 备战蓝桥杯(洛谷P1873 EKO砍树问题详解)

洛谷P1873 EKO砍树问题详解

题目描述

P1873 EKO砍树 要求在保证所有树木高度不低于H的情况下,通过砍伐使得总木材量至少为M,求最大可能的H值。

输入格式

  • 第1行:N(树木数量),M(需求木材量)
  • 第2行:N个整数表示每棵树的高度

输出格式

  • 满足条件的最大H值

算法思路

问题分析

本题核心是在离散范围内寻找最大可行解,具有典型的二分查找特征:

  1. 单调性:当H增大时,可获得的木材总量单调不增
  2. 决策可行性:对于任意H值,可在O(N)时间内验证是否满足总木材量≥M

二分法适用性

  • 搜索范围:[0, max_height](max_height为原始最高树高)
  • 目标:找到最大的H使得总木材量≥M
  • 时间复杂度:O(N log max_height),可处理1e6级数据

代码解析与优化

原始代码分析

用户提供的代码已实现核心逻辑,但存在两个优化点:

  1. 右边界冗余:直接使用2e9作为右边界,未利用输入数据特性
  2. 数组遍历方式:可从后向前遍历优化缓存命中率

优化后代码

#include <bits/stdc++.h>
using namespace std;long long N, M;
vector<long long> trees;bool check(long long H) {long long sum = 0;for (long long h : trees) {if (h > H) sum += h - H;if (sum >= M) return true;  // 提前终止优化}return sum >= M;
}int main() {ios::sync_with_stdio(false);cin.tie(nullptr);cin >> N >> M;trees.resize(N);long long max_h = 0;for (auto& h : trees) {cin >> h;max_h = max(max_h, h);}long long L = 0, R = max_h;long long ans = 0;while (L <= R) {long long mid = L + (R - L) / 2;if (check(mid)) {ans = mid;L = mid + 1;} else {R = mid - 1;}}cout << ans << endl;return 0;
}

关键优化说明

  1. 动态右边界

    long long max_h = *max_element(trees.begin(), trees.end());
    

    通过预处理获取最大树高,将二分范围缩小到实际可能区间

  2. 提前终止优化

    if (sum >= M) return true;  // Check函数内
    

    当累计木材量已满足需求时立即返回,减少不必要的计算

  3. 输入优化

    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    

    关闭C++流同步,加速大数据量输入

复杂度分析

操作时间复杂度说明
二分查找O(log max_h)典型二分复杂度
每次检查O(N)遍历所有树木计算木材量
总复杂度O(N log max_h)可处理1e6数据量(约2e7次操作)

注意事项

  1. 数据类型

    • 使用long long防止大数溢出(M可达1e18)
    • 输入输出时注意类型匹配
  2. 边界条件

    • 当M=0时,H可取最大值(需单独处理)
    • 当所有树高<H时,总木材量为0
  3. 精度问题

    • 本题H为整数,无需处理浮点精度
    • 若改为实数版本,需设置合适终止条件

测试样例

样例输入1

4 7
20 15 10 17

执行过程

  • 初始范围:[0, 20]
  • 二分过程:
    • mid=10 → sum=20+15+10+17-4*10=22 ≥7 → 尝试更大H
    • mid=15 → sum=20+17-2*15=7 ≥7 → 尝试更大H
    • mid=17 → sum=20+17-2*17=3 <7 → 回退
  • 最终结果:15

样例输出1

15

扩展思考

  1. 变种问题

    • 若要求总木材量恰好等于M,如何处理?
    • 若H可取实数,如何修改代码?
  2. 实际应用

    • 资源分配问题(如水库蓄水高度)
    • 生产调度中的最小成本问题
  3. 算法对比

    • 与三分法的区别:三分法适用于单峰函数,本题具有严格单调性

总结

本题通过二分法将看似复杂的优化问题转化为简单的可行性判断,关键点在于:

  1. 准确识别问题的单调性
  2. 高效实现可行性检查函数
  3. 合理设置二分边界

掌握这种思维模式,可以解决一类"最大最小值"问题(Maximize the Minimum Value),是算法竞赛中的重要技巧。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词