ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

cf940 E. Cashback

2022-04-24 01:31:25  阅读:143  来源: 互联网

标签:段长 长为 int cf940 Cashback 数组 2c 1e5


题意:

给定数组 a[] 和一个常数 c,可以把数组切成任意数量的段,并删除每段中前 \(\lfloor (r-l+1)/c \rfloor\) 小的数,其中分子为段长。问数组元素和的最小值。

\(n\le 1e5\)

思路:

若某段长小于 c,则可删去一个数;若段长 \([c,2c)\),则可删去两个数。

那么容易写出一种朴素dp:枚举 \(i\) 前面的所有 \(j\) 更新 \(f(i)\),根据 \(i-j\) 即最后一段的长度删数。。。明显太慢了

贪心一下,发现性质:若段长 \([c,2c)\),则可以变成一个段长为 c 的和若干段长为 1 的;若段长大于 2c,则肯定不如再切成长为 c 的段和长为 1 的段!这是因为短的段才有机会删掉更大的数。

那么就只有长为 c 的段和长为 1 的段。

\(f(i)\) 要么是 \(f(i-1)+a_i\) (当然不会是 \(f(i-2)+a_{i-1}+a_i\) 之类的,因为根据定义 \(f(i-1)\) 就是最好的)

要么把 \([i-c+1,i]\) 作为一段,即 \(f(i-c)\) 加上 \(sum[i-c+1,i]\) 减去 \([i-c+1,i]\) 中最小的 \(a_i\)。用前缀和维护区间和,并用滑动窗口维护最小值即可

const signed N = 1e5 + 3;
ll n, c, a[N], s[N], f[N];
signed main() {
    iofast;
    cin >> n >> c;
    for(int i = 1; i <= n; i++) cin >> a[i];
    for(int i = 1; i <= n; i++) s[i] = s[i-1] + a[i];

    deque<int> q;
    for(int i = 1; i <= n; i++) {
        while(q.size() && q.front() < i-c+1) q.pop_front();
        while(q.size() && a[q.back()] > a[i]) q.pop_back();
        q.pb(i);

        f[i] = f[i-1] + a[i];
        if(i >= c) f[i] = min(f[i], f[i-c]+s[i]-s[i-c]-a[q.front()]);
    }

   cout << f[n];
}

标签:段长,长为,int,cf940,Cashback,数组,2c,1e5
来源: https://www.cnblogs.com/wushansinger/p/16184232.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有