设 \(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. 关于存储最近N次数据的问题的实现

    需求描述: 需要实现采集中,始终保持最近10次的数据 描述分析: 当采集第一次数据的存储的时候,开辟一个长度为11的list,和一个标记为来记录当前的采集轮询次数(记录1-10,第11次数值归1) 数 ...

  2. StackExchange.Redis 之 Set集合 类型示例

    话不多说直接上代码: // set添加单个元素 stopwatch.Start(); "); stopwatch.Stop(); Console.WriteLine("set添加单 ...

  3. VS自定义模板-以自定义类模板为样例

    前言 在实际的工作过程中部分公司会要求开发人员在开发过程中需遵守一些开发规范,开发规范中主要包括文件的注释规范,项目.文件.变量的命名规范(例如驼峰规范)等等.例如我们代码规范中就有一项新增文件的文件 ...

  4. SqlServer视图查询效率测试

    一.  测试背景开发一个项目,数据库用的是sqlserver.帐号表数据有一两百万,不算大,也不算小.在考虑是否要使用视图,担心效率,百度了下资料,众说纷纭,好吧,实践是最好的证明,那么我们就来测试吧 ...

  5. python練習

    #登录,账户密码储存在文件中,限制登录 count = 0 for i in range(3): b = input("账号:") c = b + "," + ...

  6. java 和js 时间 格式化(yyyy-MM-dd HH:mm:ss) 以及获取当前时间

    1.js var myDate = new Date(); myDate.getYear(); //获取当前年份(2位) myDate.getFullYear(); //获取完整的年份(4位,1970 ...

  7. js获取时间及转化

    有关Javascript的本地时间获取,下面我直接用代码来进行举例吧 let time = new Date(); let month = time.getMonth()+1; //获取的月份0~11 ...

  8. Neo4j入门-开始使用

    前言 关系,指事物之间相互作用.相互影响的状态. 数据之间的关系也是如此,数据之间关系的存储在RDS就已经开始.从数据库支持的外键,到手动建立的关系表,人们采取了许多方法,只为了解决查询复杂.缓慢等问 ...

  9. 转载 selenium_对浏览器操作、鼠标操作等总结

    https://www.jianshu.com/p/7a4414082ce2 查看环境conda info --env 激活环境conda activate machine 路径改成H:cd H:\p ...

  10. linux中安装nginx时查看修改80端口时没有iptables文件的内容?? 求解