找回密码
 立即注册
查看: 10|回复: 0

2012年NOIP提高组「借教室」(洛谷P1083)解题思路与二分查找优化代码解析

[复制链接]
  • 打卡等级:即来则安
  • 打卡总天数:16
  • 打卡月天数:7
  • 打卡总奖励:111
  • 最近打卡:2025-07-12 15:59:47

21

主题

1

回帖

2469

积分

VIP会员

积分
2469
发表于 2025-7-3 15:58:17 | 显示全部楼层 |阅读模式
一、题目解读
本题为2012年NOIP提高组中的「借教室」问题(洛谷P1083),要求处理教室借用订单的分配问题。给定n天每天可用教室数量r和m个订单(订单包含需求教室数d、开始日期s、结束日期t),判断能否通过删除部分订单使所有订单满足教室容量限制。若能,输出需要删除的最少订单数;若所有订单必须保留,输出-1。题目核心在于将动态分配转化为判定性问题,通过二分查找优化求解。
二、解题思路
用户提供的代码采用二分查找解决。思路是将订单数作为二分范围,通过check函数判定是否存在可行解:
1. 转化为判定性问题:若删除前mid个订单能满足,则答案在[1, mid-1]中;否则在[mid+1, m]中。
2. 差分数组优化:用diff数组记录订单对每天教室数的增量(开始日+需求,结束日-需求),避免逐日模拟
3. 边界处理与溢出防护:使用long long防止数据溢出,特判所有订单可满足的情况,避免二分循环。
三、解题步骤
1. 输入处理:读取n、m,教室容量r和订单信息d、s、t。
2. 二分查找框架:初始化left=1, right=m,答案ans初始化为m(最坏情况全删)。
3. 判定函数check(mid):
    清空diff数组,遍历前mid个订单,按区间更新差分。
    计算每天累计需求current,若超过当日容量r,返回false。
4. 二分循环:根据check结果调整左右边界,最终ans为不可行的临界点。
5. 输出结果:特判全满足输出0,否则输出ans。
四、代码与注释
  1. #include <iostream>
  2. #include <vector>
  3. #include <cstring>
  4. using namespace std;

  5. const int MAXN = 1e6 + 10;

  6. int n, m;
  7. int r[MAXN]; // 每天可用教室数
  8. int d[MAXN], s[MAXN], t[MAXN]; // 订单信息
  9. long long diff[MAXN]; // 使用long long防止溢出

  10. bool check(int mid) {
  11.     memset(diff, 0, sizeof(diff)); // 更安全初始化
  12.     for (int i = 1; i <= mid; ++i) {
  13.         diff[s[i]] += d[i];
  14.         if (t[i] + 1 <= n) diff[t[i] + 1] -= d[i]; // 区间右端点差分
  15.     }
  16.     long long current = 0;
  17.     for (int i = 1; i <= n; ++i) {
  18.         current += diff[i];
  19.         if (current > r[i]) return false; // 超容量即不可行
  20.     }
  21.     return true;
  22. }

  23. int main() {
  24.     ios::sync_with_stdio(false);
  25.     cin.tie(0);
  26.     cin >> n >> m;
  27.     for (int i = 1; i <= n; ++i) cin >> r[i];
  28.     for (int i = 1; i <= m; ++i) cin >> d[i] >> s[i] >> t[i];
  29.     if (check(m)) { // 特判全满足
  30.         cout << 0 << endl;
  31.         return 0;
  32.     }
  33.     int left = 1, right = m, ans = m;
  34.     while (left <= right) {
  35.         int mid = (left + right) >> 1;
  36.         if (!check(mid)) {
  37.             ans = mid;
  38.             right = mid - 1;
  39.         } else {
  40.             left = mid + 1;
  41.         }
  42.     }
  43.     cout << -1 << endl << ans << endl;
  44.     return 0;
  45. }
复制代码


五、总结
该代码通过二分查找将订单分配问题转化为判定性问题,结合差分数组高效处理区间影响,避免了O(nm)的暴力复杂度。关键点在于:
1. 利用二分查找缩小答案范围,降低时间复杂度至O(mlogm)。
2. 差分数组减少冗余计算,确保判定过程线性时间。
3. 特判与溢出防护提升代码鲁棒性。
此解法适用于区间修改与查询的判定场景,是竞赛中优化动态规划的经典技巧。

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表