【题解】最大 M 子段和 Max Sum Plus Plus [Hdu1024] [51nod1052]

传送门:最大 \(M\) 子段和 \(Max\) \(Sum\) \(Plus\) \(Plus\) \([Hdu1024]\) \([51nod1052]\)

【题目描述】

给出一个长度为 \(N\) 的序列 ,将这 \(N\) 个数划分为互不相交的 \(M\) 个子段,并使得 \(M\) 个子段的和最大。

【样例】

样例输入:
7 2
-2
11
-4
13
-5
6
-2 样例输出:
26

【数据范围】

\(100\%\) \(1 \leqslant N \leqslant 10^6\)


【分析】

在 \(51nod\) 里面这道题的数据水出了一种极其诡异的境界,首先是它题目里瞎扯的一个特判没毛用,其次,我尝试了不下 \(10\) 种错误写法(包括不同题意所造成的不同写法),居然全对 \(...\)

设 \(dp[i][j]\) 表示使用前 \(j\) 个数划分了 \(i\) 段的最大和。

由于所选段不一定要连续,所以 \(dp[i][j]=dp[i][j-1]\),即第 \(i\) 段不选 \(a[j]\) 的情况。

如果要选 \(a[j]\) ,那么应该是从 \(a[j]\) 往前选出连续的一段 \([k+1,j]\) 作为第 \(i\) 段,即 \(dp[i][j]=max\{dp[i-1][k]+S[j]-S[k]]\}\),其中 \(k \in[i-1,j-1],\) 为了保证这之后剩下的数可以足够选完 \(m\) 段,需要满足 \(i-j \geqslant m-n\) 即 \(j \in [i,n-m+i]\),\(S[i]\)为 \(a[i]\) 的前缀和 。

开一个滚动数组 \(f\) 优化掉第一维,就成了酱紫: \(dp[j]=\begin{cases} max_{k=i-1}^{j-1}\{f[k]-S[k]]\}+S[j] & j=i\\max\{dp[j-1],{max_{k=i-1}^{j-1}\{f[k]-S[k]]\}+S[j]}\} & j>i \end{cases}\)

对于上面 \(j=i\) 的情况,实际操作时,可以特判,也可以直接 \(dp[i-1]=-inf\) 。

后面那一大堆求最大值的表达式可以直接用一个变量 \(maxs\) 来保存,初始值为 \(f[i-1]-S[i-1]\),每求完一个 \(dp[j]\),就让其与 \(f[j]-S[j]\) 取个最大值,下一次 \(dp[j+1]\) 要使用 \(maxs\) 时,其维护的区间信息恰好为 \([i-1,(j+1)-1]\) 。


【Code】

#include<algorithm>
#include<cstring>
#include<cstdio>
#define LL long long
#define Re register LL
using namespace std;
const int N=1e6+3;
LL n,m,tmp,ans,maxs,S[N],f[N],dp[N];
inline void in(Re &x){
int f=0;x=0;char c=getchar();
while(c<'0'||c>'9')f|=c=='-',c=getchar();
while(c>='0'&&c<='9')x=(x<<1)+(x<<3)+(c^48),c=getchar();
x=f?-x:x;
}
int main(){
while(scanf("%lld%lld",&m,&n)!=EOF){
for(Re i=1;i<=n;++i)in(S[i]),tmp+=(S[i]>0),S[i]+=S[i-1];
memset(dp,0,sizeof(dp));
memset(f,0,sizeof(f));
for(Re i=1;i<=m;++i){
for(Re j=1;j<=n;++j)f[j]=dp[j];
maxs=f[i-1]-S[i-1];
dp[i-1]=-1e18;
for(Re j=i;j<=n-m+i;++j){
dp[j]=max(dp[j-1],maxs+S[j]);
maxs=max(maxs,f[j]-S[j]);
}
}
printf("%lld\n",dp[n]);
}
}

最新文章

  1. python2.7 内置ConfigParser支持Unicode读写
  2. c++中的继承与初始化
  3. Multiple annotations found at this line
  4. js 控制框架页面跳转 top.location.herf = &quot;url&quot;
  5. 利用Native Client OLEDB 11 高效率地对SQL SERVER 进行查询和插入操作
  6. 『TCP/IP详解——卷一:协议』读书笔记——11
  7. Hadoop 在ubuntu系统上的搭建[图解]
  8. 微服务、SOA 和 API:是敌是友?
  9. java对象的内存分配
  10. N 组连续子串最大和
  11. C++算术运算符与算术表达式
  12. JSON数据表示格式简介(JavaScript对象表示法)
  13. 图片和base64相互转化
  14. .NET大批量插入数据到Oracle
  15. ASP.NET WEB API简介
  16. Oracle 增加 修改 删除 列
  17. kubernetes 简介:kube-dns 和服务发现
  18. window.frames
  19. 应用程序正常初始化(0xc0000135)失败的解决方法
  20. Python 数据类型:字典

热门文章

  1. PO设计模式
  2. yuv420 转换成 bmp
  3. Conv2D
  4. python_机器学习(一)、基本概念
  5. springboot搭建dubbo+zookeeper简单案例
  6. javascript 初探
  7. 使用Psi Probe监控Tomcat8.5
  8. 第07节-开源蓝牙协议BTStack框架代码阅读(上)
  9. LCD编程_使用调色板
  10. 高斯混合模型GMM与EM算法的Python实现