这道题实在是一道 毒瘤 题,太坑爹了。那个写 \(deque\) 的题解亲测只有80分,原因 **不言而明 **,这道题居然 丧心病狂 到 卡STL


好了,不吐槽了,进入正题

题目分析:

  • 这是一道十分 简单 的DP,相信大家也可以很容易地吧DP状态转移方程给推出来。我就献丑给大家推一遍,如有错漏,请留言,谢谢。
  • 我们可以定一个函数 \(f(i)\) ,它表示跳到 \(i\) 棵树上去的 \(f(i)\) 的劳累值。由题可知,可以 \(i-v,i-v+1,i-v+2,......i-2,i-1\) 棵树上跳到 \(i\) 棵树,所以可以推出DP状态转移方程 \(f(i)=min\{f(j)+(h(i)>=h(j))\},j\in[i-v,i)\)
  • 时间复杂度: \(\Theta(n^2)\) , 空间复杂度: \(\Theta(n)\) 。

code 1:

#include<bits/stdc++.h>
#define Maxn 1000010
#define int long long
using namespace std;
int n;
int h[Maxn];
int p;
int f[Maxn];
signed main()
{
scanf("%lld",&n);
for(int i=1;i<=n;i++)
{
scanf("%lld",&h[i]);
}
scanf("%lld",&p);
while(p--)
{
memset(f,0,sizeof(f));
int v;
scanf("%lld",&v);
f[1]=0;
for(int i=2;i<=n;i++)
{
int tmp=0x80000000;
for(int j=i-v;j<i;j++)
{
tmp=min(tmp,f[j]+(h[i]>=h[j]));
}
f[i]=tmp;
}
printf("%lld\n",f[n]);
}
return 0;
}

记录:5AC,4RE,1TLE


优化 1:

我们考虑用单调队列优化,将其优化成 \(\Theta(n)\) 。作为一个钟爱于STL的 Oler,我选择了方便而又快捷的 \(deque\) 。

我们用单调队列维护 \(f(j)+(h(i)>=h(j))\) 。维护 \(f(j)\) ,将队列中 \(f(j)\) 严格 单调递减,使得队列中的 \(f(j)\) 取出来时永远是最小的;在队尾的 \(f(j)\) 相等时,我们还要将队列中的 \(height(j)\) 进行维护,使得队列中的 \(height(j)\) 单调递增,使得队列中的 \(height(j)\) 的队首取出来永远是是最大的,使得 \((h(i)>=h(j))\) 尽量为0。

去头就简单了,假如 \(j\) 是过期的数据,即 \(j<i-v\) ,我们就把 \(j\) 给 \(pop\) 掉。


code 2:

#include<bits/stdc++.h>
#define Maxn 1000010
#define int long long
using namespace std;
int n;
int h[Maxn];
int p;
int f[Maxn];
struct node
{
int id,v;
};
deque<node>dq;
signed main()
{
scanf("%lld",&n);
for(int i=1;i<=n;i++)
{
scanf("%lld",&h[i]);
}
scanf("%lld",&p);
while(p--)
{
memset(f,0,sizeof(f));
dq.clear();
int v;
scanf("%lld",&v);
dq.push_back((node){1,0});
for(int i=2;i<=n;i++)
{
while(!dq.empty()&&dq.front().id<i-v)
{
dq.pop_front();
}
f[i]=dq.front().v+(h[i]>=h[dq.front().id]);
while(!dq.empty()&&(dq.back().v>f[i]||(dq.back().v==f[i]&&h[i]>=h[dq.back().id])))
{
dq.pop_back();
}
dq.push_back((node){i,f[i]});
}
printf("%lld\n",f[n]);
}
return 0;
}

记录:8AC,2TLE


优化 2:

题目太过于 毒瘤 ,卡掉了STL,使得STL惨遭TLE,所以我们考虑数组模拟 \(deque\) 。

定义头指针 \(head\) ,尾指针 \(tail\) ,通过移动头尾指针,来模拟 \(deque\) 。 \(pop\_front\) 即为 \(head++\) , \(pop\_back\) 即为 \(tail--\) 。我们很轻松就可以AC了。

注:

STL好用是好用,但一定要注意STL的劣势,那就是在调用函数的时候比数组慢太多。一般来说,STL不会被卡;但难保有如此一道像这一道一样的毒瘤题呢?!


code 3:

#include<bits/stdc++.h>
#define Maxn 1000010
#define int long long
using namespace std;
inline void read(int &x)
{
int f=1;x=0;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-1;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
x*=f;
}
int n;
int h[Maxn];
int p;
int f[Maxn];
struct node
{
int id,v;
};
node que[Maxn];
signed main()
{
read(n);
for(int i=1;i<=n;i++)
{
read(h[i]);
}
read(p);
while(p--)
{
memset(f,0,sizeof(f));
memset(que,0,sizeof(que));
int v;
read(v);
que[1].id=1;
que[1].v=0;
int head=1;
int tail=2;
for(int i=2;i<=n;i++)
{
while(head<tail&&que[head].id<i-v)
{
head++;
}
f[i]=que[head].v+(h[i]>=h[que[head].id]);
while(head<tail&&(que[tail].v>f[i]||(que[tail].v==f[i]&&h[i]>=h[que[tail].id])))
{
tail--;
}
que[++tail].id=i;
que[tail].v=f[i];
}
printf("%lld\n",f[n]);
}
return 0;
}

记录:10AC


最后,给你一个 \(struct\) 封装的 \(deque\) 。

code 4:

#include<bits/stdc++.h>
#define Maxn 1000010
#define int long long
using namespace std;
inline void read(int &x)
{
int f=1;x=0;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-1;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
x*=f;
}
int n;
int h[Maxn];
int p;
int f[Maxn];
int que_id[Maxn],que_v[Maxn];
struct dque
{
int head,tail;
dque()
{
head=1;
tail=0;
// memset(que,0,sizeof(que));
}
inline bool empty()
{
return head>tail;
}
inline int front_id()
{
return que_id[head];
}
inline int front_v()
{
return que_v[head];
}
inline int back_id()
{
return que_id[tail];
}
inline int back_v()
{
return que_v[tail];
}
inline void pop_front()
{
head++;
}
inline void pop_back()
{
tail--;
}
inline void push_back(int id,int v)
{
que_id[++tail]=id;
que_v[tail]=v;
}
inline void clear()
{
head=1;
tail=0;
}
}dq;
signed main()
{
read(n);
for(int i=1;i<=n;i++)
{
read(h[i]);
}
read(p);
while(p--)
{
memset(f,0,sizeof(f));
dq.clear();
int v;
read(v);
dq.push_back(1,0);
for(int i=2;i<=n;i++)
{
while(!dq.empty()&&dq.front_id()<i-v)
{
dq.pop_front();
}
f[i]=dq.front_v()+(h[i]>=h[dq.front_id()]);
while(!dq.empty()&&(dq.back_v()>f[i]||(dq.back_v()==f[i]&&h[i]>=h[dq.back_id()])))
{
dq.pop_back();
}
dq.push_back(i,f[i]);
}
printf("%lld\n",f[n]);
}
return 0;
}

记录:9AC,1TLE


如果作者有疏漏或错误的地方,请私信给我,或在评论中留言,谢谢。

洛谷P3572题解的更多相关文章

  1. [洛谷P3376题解]网络流(最大流)的实现算法讲解与代码

    [洛谷P3376题解]网络流(最大流)的实现算法讲解与代码 更坏的阅读体验 定义 对于给定的一个网络,有向图中每个的边权表示可以通过的最大流量.假设出发点S水流无限大,求水流到终点T后的最大流量. 起 ...

  2. 洛谷P5759题解

    本文摘自本人洛谷博客,原文章地址:https://www.luogu.com.cn/blog/cjtb666anran/solution-p5759 \[这道题重在理解题意 \] 选手编号依次为: \ ...

  3. 关于三目运算符与if语句的效率与洛谷P2704题解

    题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图.在每一格平原地形上最 ...

  4. c++并查集配合STL MAP的实现(洛谷P2814题解)

    不会并查集的话请将此文与我以前写的并查集一同食用. 原题来自洛谷 原题 文字稿在此: 题目背景 现代的人对于本家族血统越来越感兴趣. 题目描述 给出充足的父子关系,请你编写程序找到某个人的最早的祖先. ...

  5. 洛谷P2607题解

    想要深入学习树形DP,请点击我的博客. 本题的DP模型同 P1352 没有上司的舞会.本题的难点在于如何把基环树DP转化为普通的树上DP. 考虑断边和换根.先找到其中的一个环,在上面随意取两个点, 断 ...

  6. 【洛谷】题解 P1056 【排座椅】

    题目链接 因为题目说输入保证会交头接耳的同学前后相邻或者左右相邻,所以一对同学要分开有且只有一条唯一的通道才能把他们分开. 于是可以吧这条通道累加到一个数组里面.应为题目要求纵列的通道和横列的通道条数 ...

  7. 洛谷P3572 [POI2014]PTA-Little Bird

    P3572 [POI2014]PTA-Little Bird 题目描述 In the Byteotian Line Forest there are nn trees in a row. On top ...

  8. [洛谷P1972][题解][SDOI2009]HH的项链

    别碰我! 自己还是太蒟了…… 看了好久,最后抄参考题解打出来的…… 前面的可能影响后面的,所以按照询问右端点排序 这时候维护一个前缀和数组就可以了, 那么问题又来了,去重? 可以这样,从前往后枚举,如 ...

  9. 【洛谷P1119题解】灾后重建——(floyd)

    这道题告诉我,背的掉板子并不能解决一切问题,理解思想才是关键,比如不看题解,我确实想不清楚这题是弗洛伊德求最短路 (我不该自不量力的说我会弗洛伊德了我错了做人果然要谦虚) 灾后重建 题目背景 B地区在 ...

随机推荐

  1. putty秘钥转换成xhell支持的格式

    使用XShell导入KEY的时候报“Failed to import the user key!”错误 这个错误表明导入的private key文件不是XShell所支持的,有三种可能: 将Publi ...

  2. 使用 cxf的程序 在win10 测试部署时报空指针异常

    2018-11-08 15:50:55.072 DEBUG 21524 --- [nio-8080-exec-1] o.s.b.w.s.f.OrderedRequestContextFilter  : ...

  3. php如何定义数组常量

    是这样吗?<?php define('BEST_PHPER',array('name'=>'巩文','address'=>'china')); My God,明确告诉你不可以:原因是 ...

  4. 精通并发与 Netty (二)常用的 rpc 框架

    Google Protobuf 使用方式分析 对于 RPC 协议来说,最重要的就是对象的发送与接收,这就要用到序列化与反序列化,也称为编码和解码,序列化与反序列化和网络传输一般都在对应的 RPC 框架 ...

  5. VB非常见知识总结

    1.VB.Net设置Excel中单元格字体 sheet.Range(sheet.Cells(row, stp), sheet.Cells(row, stp)).Font.Name = "Wi ...

  6. echo-nginx-module的安装、配置、使用

    一.下载压缩包 [root@www nginx-1.16.0]# wget https://github.com/openresty/echo-nginx-module/archive/v0.61.t ...

  7. 『开发技巧』Keras自定义对象(层、评价函数与损失)

    1.自定义层 对于简单.无状态的自定义操作,你也许可以通过 layers.core.Lambda 层来实现.但是对于那些包含了可训练权重的自定义层,你应该自己实现这种层. 这是一个 Keras2.0  ...

  8. 数据结构-堆栈和队列最简单的实现(Python实现)

    OK,上篇博客我们介绍了双向链表以及代码实现,这篇文章我们来学习堆栈和队列. 队.栈和链表一样,在数据结构中非常基础一种数据结构,同样他们也有各种各样.五花八门的变形和实现方式.但不管他们形式上怎么变 ...

  9. Asp.net之实现自定义跨域

    跨域是指在浏览器的同源策略下导致前端和接口部署在不同域下导致无法直接访问的问题. 针对跨域有多种解决方案常见的有: JSNOP: 可参考Jquery实现,缺点是需要后端支持:   Access-Con ...

  10. 最大公约数GCD学习笔记

    引理 已知:k|a,k|b 求证:k|(m*a+n*b) 证明:∵ k|a ∴ 有p*k=a 同理可得q*k=b ∴ p*k*m=m*a,q*k*n=n*b ∴ k(p*m+q*n)=m*a+n*b ...