BZOJ3675: [Apio2014]序列分割(斜率优化Dp)
Description
小H最近迷上了一个分隔序列的游戏。在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列。为了得到k+1个子序列,小H需要重复k次以下的步骤:
1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的序列——也就是一开始得到的整个序列);
2.选择一个位置,并通过这个位置将这个序列分割成连续的两个非空的新序列。
每次进行上述步骤之后,小H将会得到一定的分数。这个分数为两个新序列中元素和的乘积。小H希望选择一种最佳的分割方式,使得k轮之后,小H的总得分最大。
Input
输入第一行包含两个整数n,k(k+1≤n)。
第二行包含n个非负整数a1,a2,...,an(0≤ai≤10^4),表示一开始小H得到的序列。
Output
输出第一行包含一个整数,为小H可以得到的最大分数。
Sample Input
7 3
4 1 3 4 0 2 3
4 1 3 4 0 2 3
Sample Output
108
解题思路:
这道题最重要的一点是发现最后答案是每个联通块元素和与剩余部分乘积的累和,所以和分割的顺序就无关了。
剩下的就是枚举每一位分割位置,这里相当于枚举下一刀切在哪里比较合适。
共K次,每层记录状态,发现可以斜率优化。
所以斜率优化一下就好了。
注意可以滚动数组。
代码:
1 #include<cstdio> 2 #include<cstring> 3 #include<algorithm> 4 typedef long long lnt; 5 const int N=100010; 6 int a[N]; 7 lnt s[N]; 8 lnt dp[2][N]; 9 int n,k; 10 int q[N]; 11 int h,t; 12 int cp,cq; 13 double Xx(int x) 14 { 15 return s[x]; 16 } 17 double Yy(int x) 18 { 19 return s[x]*s[x]-dp[cp][x]; 20 } 21 double Kk(int x,int y) 22 { 23 return (Yy(y)-Yy(x))/(Xx(y)-Xx(x)); 24 } 25 int main() 26 { 27 //freopen("a.in","r",stdin); 28 int nn; 29 scanf("%d%d",&nn,&k); 30 for(int i=1;i<=nn;i++) 31 { 32 n++; 33 scanf("%d",&a[n]); 34 if(a[n]==0) 35 n--; 36 } 37 for(int i=1;i<=n;i++) 38 s[i]=s[i-1]+a[i]; 39 /* if(n<=k+1) 40 { 41 lnt ans=0; 42 for(int i=1;i<=n;i++) 43 { 44 ans+=s[i]*(s[n]-s[i]); 45 } 46 printf("%lld\n",ans); 47 return 0; 48 }*/ 49 cp=1,cq=0; 50 for(int K=1;K<=k;K++) 51 { 52 cp^=cq^=cp^=cq; 53 h=1,t=1; 54 q[1]=K-1; 55 for(int i=K;i<=n;i++) 56 { 57 while(h<t&&s[i]>Kk(q[h],q[h+1])) 58 h++; 59 int j=q[h]; 60 dp[cq][i]=dp[cp][j]+s[j]*(s[i]-s[j]); 61 while(h<t&&Kk(q[t-1],q[t])>Kk(q[t],i)) 62 t--; 63 q[++t]=i; 64 } 65 } 66 printf("%lld\n",dp[cq][n]); 67 return 0; 68 }