`

【单调队列】HDU 3415 Max Sum of Max-K-sub-sequence

阅读更多

KIDx的解题报告

 

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3415

 

题意:给出一个有n个数字的环状序列(其中每个数在-1000到1000之间,且n<=100000),求一个和最大的连续子序列。(要求这个连续序列长度小于等于K)  

 

单调队列基本模型:保持队列中元素单调递增(或递减),可以两头删除,只能从队尾插入新元素,队首q[head]为当前最优区间头,队列存的是下标。

 

基本思路:由于是环,补足2*n个数, 预处理出前2*n项和,枚举区间尾i, 如果当前区间尾是i,有最优区间头q[head],那么sum[i]-sum[q[head]-1]就是区间[q[head], i]的和。

 

 

为何对于当前区间尾i最优区间头是q[head]?这就是删除队列尾的功劳了:i显然可能是当前或者以后的最优区间头,所以需要插入,而q[tail-1]是队列最后一个元素,如果区间[q[tail-1], i-1]的和小于0,要让q[tail-1]作为区间头的话,必然让和更小了,所以i是更好的区间头,于是把所有劣解q[tail-1]去掉,也就是出队列,反之q[tail-1]必然比i更优,所以说队首就是当前区间的最优的区间头了

 

#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <string>
#include <cmath>
#include <algorithm>
#include <vector>
#include <queue>
#include <map>
using namespace std;
#define M 200005
#define inf 0x3fffffff

int q[M], sum[M];

int cal (int a, int b) { return sum[b] - sum[a-1]; }

int main()
{
    int t, n, k, i;
    scanf("%d", &t);
    while (t--)
    {
        scanf ("%d%d", &n, &k);
        for (i = 1; i <= n; i++)
            scanf ("%d", sum+i);
        for (i = n+1; i <= 2*n; i++)
            sum[i] = sum[i-n];
        for (i = 2; i <= 2*n; i++)			//预处理前2*n项和(其实n+k即可)
            sum[i] += sum[i-1];
        int head = 0, tail = 0, maxs = -inf, a,  b;
        for (i = 1; i <= 2*n; i++)			//枚举区间尾i
        {
			//把劣解出队列
            while (tail > head && cal(q[tail-1], i-1) < 0) --tail;
            q[tail++] = i;

			//当前最优区间头q[head],计算区间[q[head], i]的和
            int tp = cal(q[head], i);
            if (tp > maxs)
            {
                maxs = tp;
                a = q[head];
                b = i;
            }

			//超出长度,q[head]已经不能为下一个区间尾所用就出队列
            while (tail > head && i - q[head] + 1 >= k) ++head;
        }
        if (b > n) b -= n;		//b可能大于n而不满足题意
        printf ("%d %d %d\n", maxs, a, b);
    }
    return 0;
}

 

1
4
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics