zoukankan      html  css  js  c++  java
  • APIO2010 特别行动队

    P2407 - 【APIO2010】特别行动队

    Description

    你有一支由 n 名预备役士兵组成的部队,士兵从 1 到 n 编号, 要将他们拆分成若干特别行动队调入战场。出于默契的考虑,同一支特别行动队中队员的编号应该连续,即为形如(i, i + 1, …, i + k)的序列。
    编号为 i 的士兵的初始战斗力为 xi ,一支特别行动队的初始战斗力 x 为队内士兵初始战斗力之和,即 X = Xi + Xi+1 + … + Xi+k。通过长期的观察,你总结出一支特别行动队的初始战斗力 x 将按如下经验公式修正为 x': x' = ax^2 + bx + c, 其中 a, b, c 是已知的系数( a < 0)。
    作为部队统帅,现在你要为这支部队进行编队,使得所有特别行动队修正后战斗力之和最大。 试求出这个最大和。
    例如, 你有 4 名士兵, x1 = 2, x2 = 2, x3 = 3, x4 = 4。经验公式中的参数为 a = –1,b = 10, c = –20。此时,最佳方案是将士兵组成 3 个特别行动队:第一队包含士兵1 和士兵 2,第二队包含士兵 3,第三队包含士兵 4。特别行动队的初始战斗力分别为 4, 3, 4,修正后的战斗力分别为 4, 1, 4。修正后的战斗力和为 9,没有其它方案能使修正后的战斗力和更大。

    Input

    输入由三行组成。 第一行包含一个整数 n, 表示士兵的总数。第二行包含三个整数 a, b, c, 经验公式中各项的系数。第三行包含 n 个用空格分隔的整数 x1,x2, …, xn,分别表示编号为 1, 2, …, n 的士兵的初始战斗力。

    Output

    输出一个整数,表示所有特别行动队修正后战斗力之和的最大值。

    Sample Input

    4
    -1 10 -20
    2 2 3 4

    Sample Output

    9

    Hint

    20%的数据中, n ≤ 1000;
    50%的数据中, n ≤ 10,000;
    100%的数据中, 1 ≤ n ≤ 1,000,000, –5 ≤ a ≤ –1, |b| ≤ 10,000,000, |c| ≤10,000,000, 1 ≤ xi ≤ 100。

    Source

    APIO,斜率优化 ,动态规划

     
    简单方程斜率优化,dp[i]=max(dp[j] + a*(C[i]-C[j])^2 + b*(C[i]-C[j])+c);最关键的一点,除法有精度差!!!!!!,getx()函数修改为cale(l1,l2,l3)函数,相乘计算,不用除法:
     1 #include<algorithm>
     2 //#include<iostream>
     3 #include<iomanip>
     4 #include<cstdlib>
     5 #include<cstring>
     6 #include<cstdio>
     7 #include<vector>
     8 #include<cmath>
     9 #include<queue>
    10 #include<stack>
    11 #include<map>
    12 #include<set>
    13 #define ll long long
    14 #define inf 1<<30
    15 #define rep(i,a,b) for(register int i=a;i<=b;++i)
    16 #define re register
    17 using namespace std;
    18 const int N=1000011;
    19 ll C[N],dp[N],a,b,c;
    20 int q[N];
    21 int n;
    22 inline ll gi( )
    23 {
    24     register ll ret=0,f=1;char ch=getchar();
    25     while((ch<'0'||ch>'9')&&(ch!='-')) ch=getchar();
    26     if(ch=='-') f=-1,ch=getchar();
    27     while(ch>='0'&&ch<='9') ret=ret*10+ch-'0',ch=getchar();
    28     return ret*f;
    29 }
    30 inline ll getnum(int i,int k) {
    31     register ll u=C[i]-C[k];
    32     return dp[k]+a*u*u+b*u+c;
    33 }
    34 inline bool cale(int l1,int l2,int l3) {
    35     register  ll b1=dp[l1]+a*C[l1]*C[l1]-b*C[l1],b2=dp[l2]+a*C[l2]*C[l2]-b*C[l2],b3=dp[l3]+a*C[l3]*C[l3]-b*C[l3];
    36     register ll k1=-2*a*C[l1],k2=-2*a*C[l2],k3=-2*a*C[l3];
    37     return (b3-b2)*(k1-k2) <= (b2-b1) * (k2-k3);//   ¾«¶È²î¿ÓÁËÎÒ£¡£¡£¡ 
    38 }
    39 int main( )
    40 {
    41     n=gi();a=gi(),b=gi(),c=gi();
    42     rep(i,1,n) C[i]=gi(),C[i]+=C[i-1];
    43     q[0]=0;
    44     q[1]=1;
    45     dp[0]=0;
    46     dp[1]=a*C[1]*C[1]+b*C[1]+c;
    47     register int hd=0,tl=1;
    48     rep(i,2,n) {
    49         while(hd+2<=tl&&cale(q[tl-2],q[tl-1],q[tl])) q[tl-1]=q[tl--];// bug <= 
    50         while(hd<tl&&getnum(i,q[hd])<=getnum(i,q[hd+1])) hd++;
    51         dp[i]=getnum(i,q[hd]);
    52         q[++tl]=i;
    53     }
    54     printf("%lld",dp[n]);
    55     return 0;
    56 }
  • 相关阅读:
    git常用操作
    Spring学习(5):DI的配置
    007.python学习课程(元组)
    006.python学习课程(列表)
    004.python学习课程(循环语句)
    005.python学习课程(字符串)
    003.python学习课程(条件语句)
    002.python学习课程(输入、赋值、计算)
    001.python学习课程(环境搭建、变量、数据类型、输出)
    asterisk 目录
  • 原文地址:https://www.cnblogs.com/ypz999/p/6702069.html
Copyright © 2011-2022 走看看