P2627 修剪草坪

题目描述

在一年前赢得了小镇的最佳草坪比赛后,Farm John变得很懒,再也没有修剪过草坪。现在,新一轮的最佳草坪比赛又开始了,Farm John希望能够再次夺冠。

然而,Farm John的草坪非常脏乱,因此,Farm John只能够让他的奶牛来完成这项工作。Farm John有N(1 <= N <= 100,000)只排成一排的奶牛,编号为1...N。每只奶牛的效率是不同的,奶牛i的效率为E_i(0 <= E_i <= 1,000,000,000)。

靠近的奶牛们很熟悉,因此,如果Farm John安排超过K只连续的奶牛,那么,这些奶牛就会罢工去开派对:)。因此,现在Farm John需要你的帮助,计算FJ可以得到的最大效率,并且该方案中没有连续的超过K只奶牛。

输入格式

第一行:空格隔开的两个整数 N 和 K

第二到 N+1 行:第 i+1 行有一个整数 E_i

输出格式

第一行:一个值,表示 Farm John 可以得到的最大的效率值。

输入输出样例

输入 #1

5 2
1
2
3
4
5

输出 #1

12

【思路】

单调队列 线性DP

【说在前面的话】

很有意思的一道题
正着想那是选择最多的而且区间大小不定
所以很麻烦对不对
不想做这道题目了对不对?

【前缀思路】

正着不行那就反着来!
找可以选择的奶牛很麻烦
那就找出不选的奶牛
不选的奶牛一定是小的
而且是不会和别的成一个区间
因为有两个连续的不选择的奶牛不会让结果更优
只会让结果更差
因为多选一个比少选一个会更优的
具体证明就不多说了
自己感性理解一下下就好了

【最终思路】

然后从第一个点开始枚举
入队
如果这队首的到这个点的距离大于了k + 1那就弹出队首
为什么是k + 1呢
因为这是不选择的牛
最长的连续区间是k的长度
不选择的牛就会在这个区间的两边
这种情况下是距离k + 1
并且这种情况是最长情况
所以是大于k + 1
然后因为要选择最小的嘛
那就把队尾大于这个f[i]那也弹出
因为有了比队尾更小的值那队尾就没有可能会出现了
比我小还比我强!

【注意】

要先弹出队首超出范围的值
然后再处理出f[i]的值
最后在弹出队尾的
因为f[i]的值是取决于前面合法区间内的最小值
要先把区间内不合法的弹掉
不然不能保证f[i]合法
而且弹出的比较依据就是f[i]的大小
所以先在弹出队尾之前处理出f[i]的值
这样顺序就出来了吼

【完整代码】

#include<iostream>
#include<cstdio>
#include<queue>
#define int long long
using namespace std;
const int Max = 100005;
struct node
{
    int t;
    int v;
};
deque<node>q;
int a[Max];
int f[Max];
signed main()
{
    int n,k;
    cin >> n >> k;
    int tot = 0;
    for(register int i = 1;i <= n;++ i)
        cin >> a[i],tot += a[i];
    int M = 0x7f7f7f7f7f7f;
    q.push_back((node){0,0});
    for(register int i = 1;i <= n;++ i)
    {
        while(!q.empty() && i - q.front().t > k + 1)
            q.pop_front();
        f[i] = q.front().v + a[i];
        while(!q.empty() && f[i] < q.back().v)
            q.pop_back();
        q.push_back((node){i,f[i]});
    }
    for(register int i = n;i >= n - k;i --)
        M = min(M,f[i]);
    cout << tot - M << endl;
    return 0;
}
01-16 04:14