题面

跳房子,也叫跳飞机,是一种世界性儿童游戏,也是中国民间传统的体育游戏之一。

跳房子的游戏规则如下:

 在地面上确定一个起点,然后在起点右侧画 n 个格子,这些格子都在同一条直线上。每个格子内有一个数字( 整数),表示到达这个格子能得到的分数。玩家第一次从起点开始向右跳, 跳到起点右侧的一个格子内。第二次再从当前位置继续向右跳,依此类推。

 规则规定玩家每次都必须跳到当前位置右侧的一个格子内。玩家可以在任意时刻结束游戏,获得的分数为曾经到达过的格子中的数字之和。

 现在小 R 研发了一款弹跳机器人来参加这个游戏。但是这个机器人有一个非常严重的缺陷,它每次向右弹跳的距离只能为固定的 d。小 R 希望改进他的机器人,如果他花 g 个金币改进他的机器人,那么他的机器人灵活性就能增加 g, 但是需要注意的是,每次弹  跳的距离至少为 1。 具体而言, 当g < d时, 他的机器人每次可以选择向右弹跳的距离为 d-g,d-g+1,d-g+2, …,d+g-2,d+g-1,d+g; 否则(当g ≥ d时),他的机器人每次可以选择向右弹跳的距离为1,2,3, …, d+g-2,d+g-1,d+g。

现在小 R 希望获得至少 k 分,请问他至少要花多少金币来改造他的机器人。

输入输出格式

输入格式

第一行三个正整数 n , d ,k,分别表示格子的数目,改进前机器人弹跳的固定距离,以及希望至少获得的分数。相邻两个数 之间用一个空格隔开。

接下来n行,每行两个正整数x_i,s_i ,分别表示起点到第 i 个格子的距离以及第 i 个格子的分数。两个数之间用一个空格隔开。保证 x_i​ 按递增顺序输入。

输出格式

一行,一个整数,表示至少要花多少金币来改造他的机器人。若无论如何他都无法获得至少 k分,输出 −1 。

思路

根据题意易得dp方程为f[i]=max(f[i-k...i])+s[i];

~~易得题解~~

//你发现了一个比标算还快的错误解法
#include<bits/stdc++.h>
using namespace std;
int n,d,k,ans;
long long x[500005],s[500005],f[500005];
bool check(int g)
{
    int maxl=max(d-g,1);
    int maxr=d+g;
    memset(f,-127,sizeof(f));
    f[0]=0;
    for (int i=1;i<=n;i++)
    {
        for (int j=i-1;j>=0;j--)
        {
        if (x[i]-x[j]<maxl) continue;
        if (x[i]-x[j]>maxr) break;
        f[i]=max(f[i],f[j]+s[i]);
        if (f[i]>=k) return 1;
        }
    }
    return 0;
}
int main()
{
    cin>>n>>d>>k;
    for (int i=1;i<=n;i++) cin>>x[i]>>s[i];
    int l=0,r=200005;
    while(l<=r)
    {
        int mid=(l+r)/2;
        if (check(mid)) ans=mid,r=mid-1;
        else l=mid+1;
    }
    cout<<ans<<endl;
    return 0;
}

我们知道这样是肯定过不了的,所以要想办法优化。这时我想到了单调队列的优化------*既然我们在转移中要求f[i-k...i]的最大值,那为什么不先用一个单调队列存起来然后O(1)调用呢?*

然后......题解就出来了。

注意点:过期的数(走不过来)就pop,记得开个top表示队列顶,要好做的多。

啥?为什么存编号?方便调用以及判断啊。

#include<bits/stdc++.h>
#define INF 9999999999
using namespace std;
deque<long long>q;
int n,d,k,ans=INF;
long long x[500005],s[500005],f[500005];
bool check(int g)
{
    while(!q.empty()) q.pop_back();
    int maxl=max(d-g,1);
    int maxr=d+g;
    memset(f,0,sizeof(f));
    x[0]=0;
    int top=0;
    for (int i=1;i<=n;i++)
    {
        while(x[i]>=maxl+x[top])
        {
         while(!q.empty()&&f[top]>=f[q.back()]) q.pop_back();
         q.push_back(top);
         top++;
         //puts("RUN");
        }
        while (!q.empty()&&x[i]-x[q.front()]>maxr) q.pop_front();
        if (q.empty()) f[i]=-INF;
        else f[i]=f[q.front()]+s[i];
        if (f[i]>=k) return 1;
    }
    return 0;
}
int main()
{
    cin>>n>>d>>k;
    for (int i=1;i<=n;i++) cin>>x[i]>>s[i];
    int l=0,r=2000005;
    while(l<=r)
    {
        int mid=(l+r)>>1;
        if (check(mid)) ans=mid,r=mid-1;
        else l=mid+1;
    }
    if (ans!=INF) cout<<ans<<endl;else cout<<-1<<endl;
    return 0;
}

Luogu P3957 跳房子的更多相关文章

  1. P3957 跳房子(二分答案+单调队列优化DP)

    题目链接:https://www.luogu.org/contestnew/show/4468 题目大意:跳房子,也叫跳飞机,是一种世界性的儿童游戏,也是中国民间传统的体育游戏之一. 跳房子的游戏规则 ...

  2. 洛谷 P3957 跳房子

    https://www.luogu.org/problemnew/show/P3957 错误记录:1.没开longlong 2. -inf不够小 #include<cstdio> #inc ...

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

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

  4. 洛谷P3957 跳房子 题解 二分答案/DP/RMQ

    题目链接:https://www.luogu.org/problem/P3957 这道题目我用到了如下算法: 线段树求区间最大值: 二分答案: DP求每一次枚举答案g时是否能够找到 \(\ge k\) ...

  5. P3957 跳房子

    题目描述 跳房子,也叫跳飞机,是一种世界性的儿童游戏,也是中国民间传统的体育游戏之一. 跳房子的游戏规则如下: 在地面上确定一个起点,然后在起点右侧画 n 个格子,这些格子都在同一条直线上.每个格子内 ...

  6. 洛谷P3957 跳房子(Noip2017普及组 T4)

    今天我们的考试就考到了这道题,在考场上就压根没有思路,我知道它是一道dp的题,但因为太弱还是写不出来. 下来评讲的时候知道了一些思路,是dp加上二分查找的方式,还能够用单调队列优化. 但看了网上的许多 ...

  7. 洛谷P3957 跳房子

    普及组的题.....填坑来了. 当年的我一眼二分+DP,现在都佩服起自己来了...... 然后我们就写个二分,在check里面写单调队列优化DP即可. 然后就A了...... #include < ...

  8. [luogu 3957]跳房子

    题目链接 50分做法 挺显然的一个做法,因为金币量是单调的(如果你花i枚金币可以得到最优解,i+1枚也一定可以),所以可以二分答案 然后对于二分出来的每个答案,都做一遍dp,效率$O(n^2logn) ...

  9. 2018.09.26洛谷P3957 跳房子(二分+单调队列优化dp)

    传送门 表示去年考普及组的时候失了智,现在看来并不是很难啊. 直接二分答案然后单调队列优化dp检验就行了. 注意入队和出队的条件. 代码: #include<bits/stdc++.h> ...

随机推荐

  1. poj2991 Crane(线段树)

    Description ACM has bought a new crane (crane -- jeřáb) . The crane consists of n segments of variou ...

  2. java 23种设计模式 深入浅出

    以下内容只作为对自己对知识进行总结,如有引用他人文章会在文段末尾表明出处: Java的23种设计模式 23种设计模式总共可以分为三大类,进行不定期更新总结,将逐步展开介绍自己对设计模式的理解,多多指教 ...

  3. Mobile Phone Network CodeForces - 1023F(并查集lca+修改环)

    题意: 就是有几个点,你掌控了几条路,你的商业对手也掌控了几条路,然后你想让游客都把你的所有路都走完,那么你就有钱了,但你又想挣的钱最多,真是的过分..哈哈 游客肯定要对比一下你的对手的路 看看那个便 ...

  4. Benefit UVA - 11889(已知LCM和其中一个数,求另一个数)

    首先对于C不能整除A的状况肯定排除 然后得到B=C/A 然后取G=GCD(A,B) 如果G==1,那么此时B就是解 否则的话,就证明A,B,的最小公倍数肯定不是C,因为其最小公倍数是A*B/G 那么我 ...

  5. Emacs_快捷键列表

    Emacs_快捷键列表 emacs -nw 以终端模式运行emacsLC_CTYPE=zh_CN.UTF-8 emacs C = Control M = Meta = Alt | EscAlt 可以粘 ...

  6. photoshop cc 2018破解补丁(pscc2018注册机) 附使用方法

    1.下载破解程序 破解文件自行到  http://www.ddooo.com/softdown/109954.htm 下载 博主可以到本博客的文件-->pscc2018zcj_109954.ra ...

  7. 【题解】 [ZJOI2007]矩阵游戏 (二分图匹配)

    原题目戳我 Solution: 这个二分图藏还是挺深的,重点在哪里呢?首先我们分析下,交换影响的会是哪里. 每一次交换只会影响某一行上的排列或者某一列上的排列,如果有矩阵是下面这样,就一定不会互相影响 ...

  8. COCI 2018/2019 CONTEST #2 Solution

    Problem1 Preokret 第一题一定不是什么难题. 第一个问题在读入的时候判断当前时间是不是在1440及以前就行 第二个问题考虑离线处理,由于每个时刻只能最多发生1个事件那么就弄个桶记录每一 ...

  9. 界面编程之QT的数据库操作20180801

    /*******************************************************************************************/ 一.数据库连 ...

  10. linux系统下开机启动流程

    在了解开机启动流程之前,还是得先了解一些磁盘的基本知识.磁盘主要由盘片,机械手臂,磁头,主轴马达构成.盘片就是存储数据的物理单位了.然后盘片上我们可以分成扇区(sector)和柱面(cylinder) ...