当前位置: 首页 > java >正文

每日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),是算法竞赛中的重要技巧。

http://www.xdnf.cn/news/8591.html

相关文章:

  • 几个MySQL系统调优工具
  • 黑马点评--基于Redis实现共享session登录
  • 《关于浔川社团退出DevPress社区及内容撤回的声明》
  • [C++面试] 基础题 11~20
  • 怎样改变中断优先级?
  • 酷柚易汛ERP仓储物流解决方案
  • CodeBuddy实现pdf批量加密
  • SQL注入基础
  • vue+threeJs 创造镂空管状
  • 配置tomcat时,无法部署工件该怎么办?
  • 深度学习损失“三位一体”——从 Fisher 的最大似然到 Shannon 的交叉熵再到 KL 散度,并走进 PET·P-Tuning微调·知识蒸馏的实战
  • Selenium自动化测试网页加载太慢如何解决?
  • 基于netty实现视频流式传输和多线程传输
  • 大模型的上下文context到底是啥
  • 环境搭建与工具配置
  • 博客打卡-八皇后问题
  • 用go从零构建写一个RPC(3)--异步调用+多路复用实现
  • 分布式事务知识点整理
  • ubuntu ollama /Dify/Docker部署大模型
  • 在单片机中如何在断电前将数据保存至DataFlash?
  • [docker]更新容器中镜像版本
  • Reason-ModernColBERT论文速览:Sentence- bert-基于孪生bert网络的句子嵌入
  • 【Web前端】jQuery入门与基础(一)
  • 【GESP】C++三级真题 luogu-B4039 [GESP202409 三级] 回文拼接
  • Python中tqdm进度条工具和enumerate函数的使用详解
  • 关于读取CH584单片机的IO电平出现到的乌龙
  • 从零开始:Python语言进阶之异常处理
  • vscode | Trae【实用插件】Remove empty lines 保存文件时删除空行
  • 2942. 查找包含给定字符的单词
  • 【Excel 扩展正则的能力】工作中赋予处理单元格文本的强大正则表达提取能力