Codeforces Round 894 (Div. 3) E. Kolya and Movie Theatre

本文介绍了一种使用优先队列(最小堆)解决给定整数序列中,最大子数组和问题的方法,当子数组长度受限时,通过动态调整堆来保持最大(m-1)个元素之和最大。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

对于所选择的序列a1,a2,a3,a4..aka_1,a_2,a_3,a_4..a_ka1,a2,a3,a4..ak,我们最后只需要减去k∗dk * dkd就可以了

我们可以枚举最后一个数字,根据优先队列来维护前面所有数字里,最大的m−1m-1m1个数字值之和

具体如何维护呢?

构建最小堆,只要堆的容量到了m,那就剔除该最小的数字xxx,同时将sum−=xsum-=xsum=x

#include <bits/stdc++.h>
typedef long long ll;
using namespace std;
void solve()
{
    ll n, m, d;
    cin >> n >> m >> d;
    vector<int> v(n + 1);
    for(int i = 1; i <= n; i++)
    {
        cin >> v[i];
    }
    priority_queue<int, vector<int>, greater<int>> q;
    ll sum = 0;
    ll res = 0;
    for(int i = 1; i <= n; i++)
    {
        if(v[i] >= 0)
        {
            sum += v[i];
            //超出容量 丢掉最小的
            if(q.size() == m)
            {
                sum -= q.top();
                q.pop();
            }
            q.push(v[i]);
            res = max(res, sum - i * d);
        }
    }
    cout << res << "\n";
}
int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0);
    int t;
    cin >> t;
    while (t--)
    {
        solve();
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值