码迷,mamicode.com
首页 > Windows程序 > 详细

[bzoj1911][Apio2010]特别行动队

时间:2016-10-06 19:29:54      阅读:203      评论:0      收藏:0      [点我收藏+]

标签:

Description

技术分享个元素技术分享,可以将技术分享个元素分成多组,每组的元素编号必须是连续的.

设每组的技术分享技术分享,则每组的价值公式为技术分享.

求最大价值和.

Input

输入由三行组成。

第一行包含一个整数技术分享,表示士兵的总数.

第二行包含三个整数技术分享,价值公式中各项的系数.

第三行包含技术分享个用空格分隔的整数技术分享.

Output

输出一个整数,表示最大价值和。

Sample Input

4

-1 10 -20

2 2 3 4

Sample Output

9

HINT

 技术分享

Solution

技术分享表示前技术分享个的最大价值和,

技术分享.

这样是技术分享的,显然过不了,所以考虑斜率优化.

技术分享技术分享时,

技术分享

技术分享

尽量将技术分享分离:

技术分享,

技术分享

技术分享,

技术分享.

技术分享的前提条件是

技术分享.

整理得,技术分享.

 技术分享,

则队列中的元素满足技术分享

(若存在技术分享,因为技术分享是递增的,所以技术分享技术分享优,即技术分享可以删去),

所以每次取元素时,将满足技术分享技术分享出队(因为技术分享技术分享优),然后取队首为技术分享.

#include<set> 
#include<cmath>
#include<ctime>
#include<queue>
#include<stack>
#include<cstdio>
#include<vector>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define N 1000001
using namespace std;
typedef long long ll;
ll f[N],s[N],q[N],a,b,c,n,h,t;
inline ll sqr(ll k){
    return k*k;
}
inline ll y(ll k){
    return a*sqr(s[k])+b*s[k]+c;
}
inline ll g(ll k){
    return f[k]+a*sqr(s[k])-b*s[k];
}
inline ll func(ll k){
    return a*sqr(k)+b*k+c;
} 
inline double cmp(ll p,ll q){
    return (double)(g(q)-g(p))/(double)(s[q]-s[p]);
}
inline void init(){
    scanf("%lld%lld%lld%lld",&n,&a,&b,&c);
    for(ll i=1;i<=n;i++){
        scanf("%lld",&s[i]);
        s[i]+=s[i-1];
    }
    for(ll i=1,k;i<=n;i++){
        k=(a<<1)*s[i];
        while(h<t&&cmp(q[h],q[h+1])>k) h++;
        f[i]=f[q[h]]+func(s[i]-s[q[h]]);
        while(h<t&&cmp(q[t],i)>cmp(q[t-1],q[t]))
            t--;
        q[++t]=i;
    }
    printf("%lld\n",f[n]);
}
int main(){
    freopen("commando.in","r",stdin);
    freopen("commando.out","w",stdout);
    init();
    fclose(stdin);
    fclose(stdout);
    return 0;
}

[bzoj1911][Apio2010]特别行动队

标签:

原文地址:http://www.cnblogs.com/AireenYe/p/5934065.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!