【单调队列优化dp】 分组
【单调队列优化dp】 分组
>>>>题目
【题目】
给定一行n个非负整数,现在你可以选择其中若干个数,但不能有连续k个数被选择。你的任务是使得选出的数字的和最大
【输入格式】
第一行两个整数 n,k,如题目描述
接下来一行n 个数,表示这个序列
【输出格式】
输出一行一个数,表示最大的和
【输入样例】
5 2
1 2 3 4 5
【输出样例】
12
【数据范围与约定】
对于20%的数据,保证1 <=n <=10。
对于40%的数据,保证1 <=n <=200。
对于60%的数据,保证1 <=n <=100000。
对于100%的数据,保证1 <=n <=2000000,1<=K<=n。
>>>>分析
因为题目给出非负整数,根据贪心,我们尽量多选择长度为k的序列
数据范围很大,暴力求每一段的和的做法肯定会T掉,那么考虑dp
题目涉及到求区间和,我们预处理出前缀和,用O(1)复杂度求出区间和
定义:dp[i]表示前i个人的最大收益,sum[i]表示前缀和
现在选的这一段区间和用sum[i]-sum[j]表示(i-k<=j<=i)
这段区间与 前一段长为k的区间 中间要空一个数(不能连续选k+1个数),于是我们固定i点,不断地枚举j点,找到和的最大值
因为sum求的是闭区间的前缀和 , sum[i]-sum[i]表示区间[ j+1 ,i ]的和,中间空的数就是j,再加上dp[j-1]就可以更新dp[i]的值
综上,我们可以得到状态转移方程
dp[i]=max( dp[j-1]+sum[i]-sum[j]) (i-k<=j<=i)
这里的max是对应每一个j点算出括号里的值,再求最大值
那么又怎么算最大值呢?总不能真的枚举j再找最大值吧?肯定不行,时间复杂度太高
但是你会突然发现
原方程可以拆分成 dp[i]=max(dp[j-1]-sum[j])+sum[i]
然后我们可以发现这个式子的变化与sum[i]无关,因为我们固定了i点,只是j点在变
那么我们考虑一个超棒的家伙——单调队列优化
(注意一下单调队列里面存的是下标)
定义f[j]=dp[j-1]-sum[j],将f[j]丢进单调队列里面,每次取出队首元素
扩展下一个点的时候,更新一下:f[i+1]=dp[i]-sum[i+1](和上面的式子是一样的),将它丢进单调队列里面就好啦
ヾ(๑╹◡╹)ノ"
>>>>代码
#include<bits/stdc++.h>
#define maxn 2000005
#define ll long long
using namespace std;
int head=,tail=,n,k;
ll dp[maxn],f[maxn],sum[maxn];
int a[maxn],q[maxn];
int read()
{
int x= ; char c=getchar() ;
while(c<''||c>'') c=getchar() ;
while(c>=''&&c<='') { x=x*+c-'', c=getchar() ; }
return x;
}
int main()
{
// freopen("group.in","r",stdin);
// freopen("group.out","w",stdout);
scanf("%d%d",&n,&k);
for(int i=;i<=n;i++)
{
a[i]=read();
sum[i]=sum[i-]+a[i];
}
for(int i=;i<=n;i++)
{
while(head<tail&&i-q[head]>k) head++;//如果队首的元素已经不在范围里,踢出队列
dp[i]=f[q[head]]+sum[i];//取出队首元素,更新dp[i]的值
f[i+]=dp[i]-sum[i+];//更新 f[i+1]
while(head<tail&&f[q[tail-]]<f[i+]) tail--;//维护单调队列,将小于f[i+1]的数都踢出去
q[tail++]=i+;
}
printf("%I64d",dp[n]);
return ;
}
>>>>总结
通过这道题目,我们还可以知道
对于一类dp,状态转移方程抽象为:dp[i]=max(f[j])+g[i] (l[i]<=j<=r[i])
并且l[i]~r[i]单调不减时,我们都可以考虑用单调队列优化
步骤:踢出过时元素,更新dp值,新元素入队并且维护单调队列
那么就到这里啦!
完结撒花٩(๑❛ᴗ❛๑)۶
题目来源: 2019.2.19杨雅儒学长的考试题
【单调队列优化dp】 分组的更多相关文章
- 单调队列优化DP,多重背包
单调队列优化DP:http://www.cnblogs.com/ka200812/archive/2012/07/11/2585950.html 单调队列优化多重背包:http://blog.csdn ...
- bzoj1855: [Scoi2010]股票交易--单调队列优化DP
单调队列优化DP的模板题 不难列出DP方程: 对于买入的情况 由于dp[i][j]=max{dp[i-w-1][k]+k*Ap[i]-j*Ap[i]} AP[i]*j是固定的,在队列中维护dp[i-w ...
- hdu3401:单调队列优化dp
第一个单调队列优化dp 写了半天,最后初始化搞错了还一直wa.. 题目大意: 炒股,总共 t 天,每天可以买入na[i]股,卖出nb[i]股,价钱分别为pa[i]和pb[i],最大同时拥有p股 且一次 ...
- Parade(单调队列优化dp)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=2490 Parade Time Limit: 4000/2000 MS (Java/Others) ...
- BZOJ_3831_[Poi2014]Little Bird_单调队列优化DP
BZOJ_3831_[Poi2014]Little Bird_单调队列优化DP Description 有一排n棵树,第i棵树的高度是Di. MHY要从第一棵树到第n棵树去找他的妹子玩. 如果MHY在 ...
- [小明打联盟][斜率/单调队列 优化dp][背包]
链接:https://ac.nowcoder.com/acm/problem/14553来源:牛客网 题目描述 小明很喜欢打游戏,现在已知一个新英雄即将推出,他同样拥有四个技能,其中三个小技能的释放时 ...
- 单调队列以及单调队列优化DP
单调队列定义: 其实单调队列就是一种队列内的元素有单调性的队列,因为其单调性所以经常会被用来维护区间最值或者降低DP的维数已达到降维来减少空间及时间的目的. 单调队列的一般应用: 1.维护区间最值 2 ...
- BZOJ1791[Ioi2008]Island 岛屿 ——基环森林直径和+单调队列优化DP+树形DP
题目描述 你将要游览一个有N个岛屿的公园.从每一个岛i出发,只建造一座桥.桥的长度以Li表示.公园内总共有N座桥.尽管每座桥由一个岛连到另一个岛,但每座桥均可以双向行走.同时,每一对这样的岛屿,都有一 ...
- P4381 [IOI2008]Island(基环树+单调队列优化dp)
P4381 [IOI2008]Island 题意:求图中所有基环树的直径和 我们对每棵基环树分别计算答案. 首先我们先bfs找环(dfs易爆栈) 蓝后我们处理直径 直径不在环上,就在环上某点的子树上 ...
随机推荐
- 利用BLEU进行机器翻译检测(Python-NLTK-BLEU评分方法)
双语评估替换分数(简称BLEU)是一种对生成语句进行评估的指标.完美匹配的得分为1.0,而完全不匹配则得分为0.0.这种评分标准是为了评估自动机器翻译系统的预测结果而开发的,具备了以下一些优点: 计算 ...
- RDLC报表数据集的一个细节,导致错误为 尚未数据源提供数据源实例
报表中,数据集的名字DataSet_CZ, 这里报表这样加载,视乎是的. reportViewer1.LocalReport.DataSources.Add(new Microsoft.Reporti ...
- 解决SVN提交和更新代码冲突?
解决冲突有三种选择: 1.放弃自己的更新,使用svn revert(回滚),然后提交.在这种方式下不需要使用svn resolved(解决) 2.放弃自己的更新,使用别人的更新.使用最新获取的版本覆盖 ...
- php读取文件使用redis的pipeline(管道)导入大批量数据
需求:需要做一个后台上传TXT文件,读取其中的内容,然后导入redis库中.要求速度快,并且支持至少10W以上的数据,而内容也就一个字段存类似openid和QQ 传统做法:我一开始做的时候就老套路,遍 ...
- CCF CSP 201409-1 相邻数对
题目链接:http://118.190.20.162/view.page?gpid=T16 问题描述 试题编号: 201409-1 试题名称: 相邻数对 时间限制: 1.0s 内存限制: 256.0M ...
- shell编辑器vi的常用命令
一:翻页 ctrl+u向上翻半页 ctrl+f向上翻一页 ctrl+d 向下翻半页 ctrl+b 向下翻一页 二:移动光标指令 0: 光标移至当前行首 $: 光标移至当前行尾 三:常用插入.删除指令 ...
- HL-D793G七位蓝色+PT6311驱动
和之前那块一起买的.又画了这个屏幕驱动,还画了灯丝电源驱动,和以前画的stm8核心板在JLC拼板. 亚光黑颜色,不单独收钱! 切好水洗吹干! 忘记给排针加丝印,手写代之. 焊接的时候,脑子发热,以为把 ...
- mycat 单库多表实现水平分片
环境 mycat : 192.168.126.128 root root mysql1: 192.168.126.129:3306 root lizhenghua mysql2: 192.168.12 ...
- sql的优化
适当的索引 尽量不要有空判断的语句 尽量不要使用不等于条件 尽量不要使用or条件 尽量不要使用左右模糊查询 尽量使用exists代替in 尽量避免一次性返回大数据量
- 编程类-----matlab基础语法复习(2)
2019年美赛准备:matlab基本题目运算 clear,clc %% 计算1/3 + 2/5 + ...3/7 +10/21 % i = 1; j = 3; ans = 0; % while i & ...