题面

传送门

分析

显然答案有单调性,可以二分答案,设当前二分值为g,根据题意我们可以求出跳跃长度的范围[l,r]

考虑DP

子状态: dp[i]表示跳到第i个点时的最大和

状态转移方程 \(dp[i]=max(dp[i],dp[j]+a[i]) (j \in [1,n),x[i]-x[j] \in [l,r])\)

初始值:dp[0]=0 (把起点看成第0号点,权值和坐标都为0)

直接转移的时间复杂度是\(O(n^2)\)

由于此题数据水,\(O(n^2logn)\)可以卡过

(相信热爱学习,不屑于打暴力的你一定会继续往下看的)

观察状态转移方程,发现满足条件的j一定在某个区间内,且区间在不断移动,类似“滑动窗口问题”,可建立一个单调队列

另外,j存在决策单调性,即i增加时,j一定也不断增加,不会再减小(例如对于i=1时我们求出满足条件的最大j,i=2时满足条件的j一定比i=1时的j更大)

因此,每次循环时j不必重置成0,这样可显著减少时间

然后维护单调队列

对于每个i,我们将满足条件(与i距离>l)的dp[j]不断加入队尾,并同时维护序列的单调性,保证队列从大到小递减,这样队头的答案一定最优

接着处理不合格的情况,即弹出队头与i距离>r的值

现在队头就是满足\(j \in [1,n),x[i]-x[j] \in [l,r]\)的最大dp[j]+a[i],因此用队头更新dp[i],i++,继续下一个循环

DP时间复杂度\(O(n)\)

总时间复杂度\(O(nlogn)\)

代码

暴力卡过:

#include<iostream>
#include<cstdio>
#include<cstring>
#define maxn 500005
#define INF 0x3f3f3f3f3f3f3f3f
using namespace std;
inline int qread() {
int x=0,sign=1;
char c=getchar();
while(c<'0'||c>'9') {
if(c=='-') sign=-1;
c=getchar();
}
while(c>='0'&&c<='9') {
x=x*10+c-'0';
c=getchar();
}
return x*sign;
}
int n,d;
int k;
int x[maxn];
int a[maxn];
long long dp[maxn];
int check(int g) {
int l0,r0;
memset(dp,-0x3f,sizeof(dp));
long long ans=dp[0];
if(g<d) {
l0=d-g;
r0=d+g;
} else {
l0=1;
r0=d+g;
}
dp[0]=0;
for(int i=1; i<=n; i++) {
for(int j=i-1; j>=0; j--) {
if(x[i]-x[j]<l0) continue;
if(x[i]-x[j]>r0) break;
if(dp[j]+a[i]>dp[i]) dp[i]=dp[j]+a[i];
if(dp[i]>=k) return 1; }
if(dp[i]>ans) ans=dp[i];
}
if(ans>=k) return 1;
else return 0;
} int main() {
n=qread();
d=qread();
k=qread();
for(int i=1; i<=n; i++) {
x[i]=qread();
a[i]=qread();
}
dp[0]=0;
x[0]=0;
int l=0,r=1005;
int mid;
int ans=-1;
int t;
while(l<=r) {
mid=(l+r)>>1;
if(check(mid)) {
ans=mid;
r=mid-1;
} else l=mid+1;
}
printf("%d\n",ans);
}
//Dedicated to Selina

单调队列:

#include<iostream>
#include<cstdio>
#include<cstring>
#define maxn 500005
#define INF 0x3f3f3f3f3f3f3f3f
using namespace std;
int n,d;
int k;
int x[maxn];
int a[maxn];
long long dp[maxn];
struct node{
long long v;
int x;
node(){ }
node(long long val,int pos){
v=val;
x=pos;
}
};
struct deque{
int head,tail;
node Q[maxn];
node front(){
return Q[head];
}
node back(){
return Q[tail-1];
}
void push_back(node p){
Q[tail]=p;
tail++;
}
void pop_front(){
head++;
}
void pop_back(){
tail--;
}
bool empty(){
if(head<tail) return 0;
else return 1;
}
deque(){
head=tail=0;
}
void clear(){
head=tail=0;
}
}q;
int check(int g){
int l0,r0;
if(g<d) {
l0=d-g;
r0=d+g;
} else {
l0=1;
r0=d+g;
}
int j=0;
memset(dp,-0x3f,sizeof(dp));
q.clear();
dp[0]=0;
for(int i=1;i<=n;i++){
while(x[i]-x[j]>=l0){//由于j有决策单调性,不必清零
while(!q.empty()&&dp[j]>=q.back().v) q.pop_back();//保证序列单调递减
q.push_back(node(dp[j],x[j]));
j++;
}
while(!q.empty()&&x[i]-q.front().x>r0) q.pop_front();//排除不符合条件的情况
if(q.empty()) dp[i]=-INF;//如果队列为空,说明该点不能到达,直接设为-INF
else dp[i]=q.front().v+a[i];
if(dp[i]>=k) return 1;
}
return 0;
}
int main(){
scanf("%d %d %d",&n,&d,&k);
for(int i=1;i<=n;i++){
scanf("%d %d",&x[i],&a[i]);
}
x[0]=0;
int l=0,r=x[n];
int mid;
int ans=-1;
while(l<=r){
mid=(l+r)>>1;
if(check(mid)){
ans=mid;
r=mid-1;
}else l=mid+1;
}
printf("%d\n",ans);
}
//Dedicated to Selina

luogu P3657 (NOIP2017) 跳房子(二分+DP+单调队列)的更多相关文章

  1. 洛谷 P3957 跳房子 —— 二分答案+单调队列优化DP

    题目:https://www.luogu.org/problemnew/show/P3957 先二分一个 g,然后判断: 由于转移的范围是一个区间,也就是滑动窗口,所以单调队列优化: 可以先令队尾为 ...

  2. [acmm week12]二分+dp+单调队列

    1004 抄作业         Time Limit: 1sec    Memory Limit:256MB Description Zfree虽然平时很爱学习,但是他迫于生活所迫(比如设计cpu实 ...

  3. bzoj 2806: [Ctsc2012]Cheat【广义SAM+二分+dp+单调队列】

    把模板串建一个广义SAM 然后在线查询,每次在SAM上预处理出一个a[i]表示i位置向前最多能匹配多长的模板串 二分答案L,dp判断,设f[i]为·~i有几个匹配,转移显然是f[i]=max{f[i- ...

  4. 【BZOJ 2806】 2806: [Ctsc2012]Cheat (SAM+二分+DP+单调队列)

    2806: [Ctsc2012]Cheat Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 1262  Solved: 643 Description ...

  5. [NOIP2017普及组]跳房子(二分,单调队列优化dp)

    [NOIP2017普及组]跳房子 题目描述 跳房子,也叫跳飞机,是一种世界性的儿童游戏,也是中国民间传统的体育游戏之一. 跳房子的游戏规则如下: 在地面上确定一个起点,然后在起点右侧画 nn 个格子, ...

  6. [BZOJ 2500]幸福的道路 树形dp+单调队列+二分答案

    考试的时候打了个树链剖分,而且还审错题了,以为是每天找所有点的最长路,原来是每天起点的树上最长路径再搞事情.. 先用dfs处理出来每个节点以他为根的子树的最长链和次长链.(后面会用到) 然后用类似dp ...

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

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

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

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

  9. BZOJ_3316_JC loves Mkk_ 二分答案 + 单调队列

    BZOJ_3316_JC loves Mkk_ 二分答案 + 单调队列 题意: 分析: 拆成链,二分答案,奇偶两个单调队列维护最大子段和,记录方案. 代码: #include <cstdio&g ...

随机推荐

  1. 零点.Net Core 接触

    一.Program.cs类与Startup类 1.一切从Main开始,Main方法包含了是整个应用程序的入口 ASP.NET Core应用程序可以配置和启动主机(Host). 主机负责应用程序启动和生 ...

  2. wireshare文件格式

    你用Wireshark打开这个pkt试试,如果可以打开,就说明Wireshark支持这种格式.然后你就可以去看Wireshark的源码.*.pkt是omnipeek/etherpeek的默认文件格式, ...

  3. ssm框架整合抽取BaseDao接口

    import java.io.Serializable; import java.util.List; /** * DAO基础操作模板 * * @param <T> 泛型 */ publi ...

  4. springboot 集成rabbitMQ

    package com.jd.ng.shiro.config.rabbitMQconfig; import com.jd.ng.shiro.rabbitMqListener.SimpleMessage ...

  5. oracle 11g 执行先决条件检查失败的解决方法

    在安装oracle 11g时,出现执行先决条件失败的情况如下: 你可以忽略所有强制安装,一般不会影响功能,但如果你想知道为什么会产生这种错误, 并且当出现以上情况时又该如何解决呢?如下列出了原因和解决 ...

  6. Django中如何将javascript中的变量传给位于javascript内的{% url %}中的参数?

    这个问题困扰了我一天,不知道困扰了你多久,希望能帮助你 django 中的url模版使用起来可以说是非常方便的,但是怎么在url模版中传入参数的 {% url "url.index" ...

  7. django 修改字段后,同步数据库,失败:django.db.utils.InternalError: (1054, "Unknown column 'api_config.project_id_id' in 'field list'")

    问题原因是,修改字段后,同步失败了,然后执行查询的时候,就会提示这个错误,这个字段没有 最暴力的方法可以直接在数据库中修改字段,但是修改后,models没同步,可能会存在问题,因此开始我的百度之旅(这 ...

  8. Jenkins插件--通知Notification

    参考来源:http://blog.csdn.net/wangmuming/article/details/22925357 ============================ 题外话 邮箱配置需 ...

  9. 【leetcode】1090. Largest Values From Labels

    题目如下: We have a set of items: the i-th item has value values[i] and label labels[i]. Then, we choose ...

  10. python之requests模块中的params和data的区别

    params的时候之间接把参数加到url后面,只在get请求时使用: import requests url='https://api.ireaderm.net/account/charge/info ...