AcWing 1090. 绿色通道

\(AcWing\) \(1090\). 绿色通道

一、题目描述

高二数学《绿色通道》总共有 \(n\) 道题目要抄,编号 \(1,2,…,n\),抄第 \(i\) 题要花 \(a_i\) 分钟。

\(Y\) 决定只用 不超过 \(t\) 分钟 抄这个,因此必然有空着的题。

每道题要么不写,要么抄完,不能写一半。

下标连续的一些空题称为一个空题段,它的长度就是所包含的题目数。

这样应付自然会引起马老师的愤怒,最长的空题段越长,马老师越生气。

现在,小 \(Y\) 想知道他在这 \(t\) 分钟内写哪些题,才能够尽量减轻马老师的怒火。

由于小 \(Y\) 很聪明,你只要 告诉他最长的空题段至少有多长 就可以了,不需输出方案。

输入格式
第一行为两个整数 \(n,t\)

第二行为 \(n\) 个整数,依次为 \(a_1,a_2,…,a_n\)

输出格式
输出一个整数,表示最长的空题段至少有多长。

数据范围
\(0<n≤5×10^4\),\(0<a_i≤3000\),\(0<t≤10^8\)

输入样例:

17 11
6 4 5 2 5 3 4 5 2 3 4 5 2 3 6 3 5

输出样例

3

二、二分+\(DP\)

\(t<=10^8\),好\(BT\)的数据上限!这就是在提示我们,不能选择\(O(N)\)的算法,那样基本上凉了!需要有一个\(O(logN)\)的算法: 二分

目标

  • \(t\) 分钟内写完作业
  • 尽量减轻马老师的怒火
  • 找出最长空白长度

分析

  • 空白区越大,用时越少
  • 空白区越小,用时越大

如果我们选择 二分 的思路:

通过二分,找出一个当前的尝试的区间空白最大值\(limit\),

如果当前枚举到的区间空白最大值\(limit\),

  • 可以在\(t\)分钟内完成,那么让\(limit\)再小一点试试,如果小一点的也可以在\(t\)分钟内完成,马老师的怒火会更小一点,惹毛了马老师,后果严重。
  • 不能在\(t\)分钟内完成,那么让\(limit\)再大一点试试,这样也许就能写完了,马老师肯定会更生气,但这不是最重要的,最重要的是我得能在\(t\)分钟内写完。

当这样不断的二分查找到最后时,就达到了一种 状态

  • 能在\(t\)分钟内完成
  • 尽量让空白区间最大值越小
  • 在能完成作业的情况下,尽量多的写一些,马老师的愤怒值最小

2、细节问题
如果\(i\)选中,那么它前面最远的可以是哪个?此处与烽火传递不太一样,那个是 在连续 \(m\) 个烽火台中至少要有一个发出信号,翻译一下就是连续\(m\)个中必须有一个,也就是空白区最多是\(m-1\)个,此题是空白区是\(limit\)个,比那个大\(1\)

\(Code\)

#include <bits/stdc++.h>

using namespace std;
const int N = 50010;
const int INF = 0x3f3f3f3f;

int n, m;
int w[N];
int f[N]; // 已经完成前i个,并且第i个被选中 情况下抄袭时间最小值

// 暴力+DP 通过了 8/12个数据
bool check(int limit) {
    // 注意这个初始化,我吃了这个亏~
    memset(f, 0x3f, sizeof f);
    f[0] = 0;

    for (int i = 1; i <= n; i++)                             // 计算前i道题,填表填数据
        for (int j = max(0, i - limit - 1); j <= i - 1; j++) // 第i题肯定是选中的,那么它前面的是哪个位置选中呢?
            f[i] = min(f[i], f[j] + w[i]);

    // 最后[n-limit,n]之间必然得有一题答了吧,这些都是合法解
    int res = INF;
    for (int i = n - limit; i <= n; i++) res = min(res, f[i]);
    // 最小值比m小的话,那么就是有可行解
    return res <= m;
}

int main() {
    cin >> n >> m;
    for (int i = 1; i <= n; i++) cin >> w[i];

    // 最长的空题段∈[1,n]

    // 二分(求最长的最小)
    int l = 0, r = n; // 空段区间长度0~n都有可能,注意此处是带0的,因为可能老师要求每题必答,对吧,你以为隔一题空一题老师能忍,其实人家忍不了!
    // 事实也证明了这一点,我尝试修改为 int l=1,r=n;会挂掉一个测试点! 
    /*
    空白时段的长度越长,花费的时间越少
    空白时段的长度越短,花费的时间越长
    */
    while (l < r) {
        int mid = (l + r) >> 1; // 一定要搞清楚mid的含义:模拟的最长空题段长度
        if (check(mid))         // 如果这个空白长度可以使得整体时间<=m,那么空白长度可以再小一点,这是个弯,好好想想
            r = mid;
        else
            l = mid + 1; // 如果现在的长度不能使得时间够用,只能是长度再长一点
    }
    printf("%d\n", l);
    return 0;
}

三、单调队列优化

上面的思路是对的,因为可以正常通过\(12\)中的\(7\)个测试点,但其它的\(TLE\)了,需要优化:

因为状态转移方程定义的方式,决定了\(i\)是必选的,那么\(w[i]\)确定 要忍受的代价,整体代价要想最小,只能是前面的代价最小!

\(f[i]\)的所有前序可选状态是:\(i\)左侧\(limit+1\)个范围,也就是要找出这个范围内的\(min(f[j])\),\(j \in [i-limit-1,i-1]\)

指定区间内的最小值问题,可以使用单调队列来优化:

\(Q\):答案在哪里?

需要在\(n \sim n-limit\)之间去枚举找出最小值做为答案。

举个栗子秒懂:
\(n=10,limit=3\),那么如果\(7\)被选中,那么\(8,9,10\)是不用写的。

#include <bits/stdc++.h>

using namespace std;
const int N = 50010;
const int INF = 0x3f3f3f3f;

int n, m;
int w[N];
int f[N];
int q[N], hh, tt;

bool check(int limit) {
    hh = 0, tt = 0, q[0] = 0;
    for (int i = 1; i <= n; i++) {
        // 窗口的长度是limit
        while (hh <= tt && q[hh] < i - limit - 1) hh++;

        // 直接计算
        f[i] = f[q[hh]] + w[i];

        // i入队列
        while (hh <= tt && f[q[tt]] >= f[i]) tt--;
        q[++tt] = i;
    }
    // 答案可能存在于 n,n-2,...n-m中,枚举一下求最少值即可
    int res = INF;
    for (int i = n - limit; i <= n; i++) res = min(res, f[i]);
    // 最小值比m小的话,那么就是有可行解,我才不管是不是有其它的值也比m小不小!
    return res <= m;
}

int main() {
    cin >> n >> m;
    for (int i = 1; i <= n; i++) cin >> w[i];
    // 二分(求最长的最小),就是碰一下区间的长度,大了就变小,小了就变大
    int l = 0, r = n; // 空段区间长度0~n都有可能
    // 当我们允许的空白时段的长度越长,那么花费的时间越少
    while (l < r) {
        int mid = (l + r) >> 1;
        if (check(mid))
            r = mid;
        else
            l = mid + 1;
    }
    cout << l << endl;
    return 0;
}
posted @ 2022-01-18 13:34  糖豆爸爸  阅读(103)  评论(2编辑  收藏  举报
Live2D