设 \(f[i][j]\) 为第 \(i\) 天在第 \(j\) 个位置放置的最大值,设 \(s[i][j]\) 是第 \(i\) 行的前缀和,则

\[\begin{align}
f[i][j] & =s[i+1][j+k-1]-s[i+1][j-1]+ \\ \max_l &
\begin{cases}
f[i-1][l]+s[i][j+k-1]-s[i][j-1] & (1 \leq l \leq j-k) \\
f[i-1][l]+s[i][j+k-1]-s[i][l+k-1] & (j-k+1 \leq l \leq j) \\
f[i-1][l]+s[i][l-1]-s[i][j-1] & (j+1 \leq l \leq j+k-1) \\
f[i-1][l]+s[i][j+k-1]-s[i][j-1] & (j+k \leq l \leq m-k+1)
\end{cases}
\end{align}
\]

如果暴力转移,则复杂度 \(O(nm^2 )\)

如果 \(k\) 很小,那么对中间两种情况暴力转移,旁边两种由于只有 \(f[i-1][l]\) 与 \(l\) 有关,可以预处理前后缀 \(\max\) 来解决,复杂度 \(O(nmk)\)

当 \(k\) 变大时,两侧的情况仍然暴力转移,中间的情况可以暴力用以 \(l\) 为下标的单调队列维护 \(f[i-1][l]-s[i][l+k-1]\) 和 \(f[i-1][l]+s[i][l-1]\)

(如果想偷懒也可以敲个线段树维护一下)

(发现单调队列优化DP不太熟练,准备要复习下)

#include <bits/stdc++.h>
using namespace std; #define int long long
const int N = 55, M = 20005; int a[N][M],s[N][M],f[N][M],n,m,k,q[M],qt[M],l,r; signed main() {
cin>>n>>m>>k;
for(int i=1;i<=n;i++) {
for(int j=1;j<=m;j++) {
cin>>a[i][j];
s[i][j]=s[i][j-1]+a[i][j];
}
}
for(int i=1;i<=m-k+1;i++)
f[1][i]=s[1][i+k-1]-s[1][i-1]+s[2][i+k-1]-s[2][i-1];
for(int i=2;i<=n;i++) {
for(int j=1;j<=m-k+1;j++)
q[j]=f[i-1][j];
for(int j=1;j<=m-k+1;j++)
q[j]=max(q[j],q[j-1]);
for(int j=k;j<=m-k+1;j++)
f[i][j]=max(f[i][j],q[j-k]+s[i][j+k-1]-s[i][j-1]);
for(int j=m-k+1;j;j--)
q[j]=f[i-1][j];
for(int j=m-k+1;j;j--)
q[j]=max(q[j],q[j+1]);
for(int j=1;j<=m-k+1;j++)
f[i][j]=max(f[i][j],q[j+k]+s[i][j+k-1]-s[i][j-1]);
l=1;r=0;
for(int j=1;j<=m-k+1;j++) q[j]=-1e9;
for(int j=1;j<=m-k+1;j++) {
while(l<=r && q[r]<f[i-1][j]-s[i][j+k-1]) --r;
++r;
q[r]=f[i-1][j]-s[i][j+k-1];
qt[r]=j;
while(l<=r && qt[l]<j-k+1) ++l;
if(l<=r) f[i][j]=max(f[i][j],q[l]+s[i][j+k-1]);
}
l=1;r=0;
for(int j=1;j<=m-k+1;j++) q[j]=-1e9;
for(int j=m-k+1;j;--j) {
while(l<=r && q[r]<f[i-1][j]+s[i][j-1]) --r;
++r;
q[r]=f[i-1][j]+s[i][j-1];
qt[r]=j;
while(l<=r && qt[l]>j+k-1) ++l;
if(l<=r) f[i][j]=max(f[i][j],q[l]-s[i][j-1]);
}
for(int j=1;j<=m-k+1;j++) f[i][j]+=s[i+1][j+k-1]-s[i+1][j-1];
}
int ans=0;
/*for(int i=1;i<=n;i++) {
for(int j=1;j<=m;j++) cout<<f[i][j]<<"\t";
cout<<endl;
}*/
for(int i=1;i<=m;i++) ans=max(f[n][i],ans);
cout<<ans;
}

[CF1304F] Animal Observation - dp,单调队列的更多相关文章

  1. Codeforces 1304F1/F2 Animal Observation(单调队列优化 dp)

    easy 题目链接 & hard 题目链接 给出一张 \(n \times m\) 的矩阵,每个格子上面有一个数,你要在每行选出一个点 \((i,t)\),并覆盖左上角为 \((i,t)\), ...

  2. [poj3017] Cut the Sequence (DP + 单调队列优化 + 平衡树优化)

    DP + 单调队列优化 + 平衡树 好题 Description Given an integer sequence { an } of length N, you are to cut the se ...

  3. DP+单调队列 codevs 1748 瑰丽华尔兹(还不是很懂具体的代码实现)

    codevs 1748 瑰丽华尔兹 2005年NOI全国竞赛  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 大师 Master 题解       题目描述 Descripti ...

  4. 习题:烽火传递(DP+单调队列)

    烽火传递[题目描述]烽火台又称烽燧,是重要的防御设施,一般建在险要处或交通要道上.一旦有敌情发生,白天燃烧柴草,通过浓烟表达信息:夜晚燃烧干柴,以火光传递军情.在某两座城市之间有n个烽火台,每个烽火台 ...

  5. (noip模拟二十一)【BZOJ2500】幸福的道路-树形DP+单调队列

    Description 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光. 他们画出了晨练路线的草图,眼尖的小T发现可以用树来描绘这个草图. ...

  6. 3622 假期(DP+单调队列优化)

    3622 假期 时间限制: 1 s 空间限制: 64000 KB 题目等级 : 黄金 Gold 题目描述 Description 经过几个月辛勤的工作,FJ决定让奶牛放假.假期可以在1-N天内任意选择 ...

  7. bzoj2500: 幸福的道路(树形dp+单调队列)

    好题.. 先找出每个节点的树上最长路 由树形DP完成 节点x,设其最长路的子节点为y 对于y的最长路,有向上和向下两种情况: down:y向子节点的最长路g[y][0] up:x的次长路的g[x][1 ...

  8. URAL 1427. SMS(DP+单调队列)

    题目链接 我用的比较传统的办法...单调队列优化了一下,写的有点搓,不管怎样过了...两个单调队列,存两个东西,预处理一个标记数组存... #include <iostream> #inc ...

  9. [USACO2004][poj2373]Dividing the Path(DP+单调队列)

    http://poj.org/problem?id=2373 题意:一条直线分割成N(<=25000)块田,有一群奶牛会在其固定区域吃草,每1把雨伞可以遮住向左右延伸各A到B的区域,一只奶牛吃草 ...

随机推荐

  1. MySQL 什么是索引?

    该文为< MySQL 实战 45 讲>的学习笔记,感谢查看,如有错误,欢迎指正 一.索引简介 索引就类似书本的目录,作用就是方便我们更加快速的查找到想要的数据. 索引的实现方式比较多,常见 ...

  2. kubernetes 资源管理

    前言 在kubernetes环境下,无论集群再大,对应的集群资源(cpu.memory.storage)总是有上限的.而默认情况下,我们启动的pod.以及pod中运行的容器,对应的资源是不加限制的.理 ...

  3. 使用ffmpeg为影片添加字幕

    ffmpeg -i infile.mp4 -i infile.srt -c copy -c:s mov_text outfile.mp4 The order of -c copy -c:s mov_t ...

  4. hive内置方法一览

    引用 https://www.cnblogs.com/qingyunzong/p/8744593.html#_label0 官方文档 https://cwiki.apache.org/confluen ...

  5. Java学习之String、StringBuffer、StringBuilder

    String 我们知道字符串的分配和其他对象分配一样,是需要消耗高昂的时间和空间的,而且字符串我们使用的非常多.JVM为了提高性能和减少内存的开销,在实例化字符串的时候进行了一些优化:使用字符串常量池 ...

  6. .NET Core DI简单介绍

    依赖注入的基础 我们都知道DI,,他是一种开发模式.他是将服务可被应用程序任何位置的代码使用,当某个代码组件(如一个类)需要引用某些外部代码(一个服务)时.都有两种选择 1:直接在调用代码种创建服务组 ...

  7. JAVA全套资料含视频源码(持续更新~)

    本文旨在免费分享我所搜集到的Java学习资源,所有资源都是通过正规渠道获取,不存在侵权.现在整理分享给有所需要的人. 希望对你们有所帮助!有新增资源我会更新的~大家有好的资源也希望分享,大家互帮互助共 ...

  8. Essential C++ 笔记-1

    本文作者为C++初学者,学习之中难免有误,该文章仅为参考 面向对象概述 继承:改变类之间的关系 多态:让基类的pointer或refence得以十分透明的指向基类的某个派生对象 继承 继承发生在对象与 ...

  9. Mybaits(10)N+1问题

    N+1问题 从上面的例子日志中我们可以看到所有级联都成功了,但是引发了性能问题,例如我们在查询雇员的信息和工作任务信息,此时体检表和工牌信息就是多余,我们没必要查询一次.如果想日志体现的那样,取出了所 ...

  10. html font后面跟多种字体

    例子 body,div,dl,dt,dd,li,h1,h2,h6,pre,form,fieldset,input,textarea,p,blockquote,th,td { margin: 0; pa ...