【算法学习笔记】RMQ问题与ST表
RMQ问题
人话翻译
给定一个长度为的数列,然后有组询问,每次询问一个区间的最小值。
其中
暴力做法
很显然,暴力做法就是便历 。这个做法最坏时间复杂度将会高达。很显然,这对于的数据范围要炸
正解
线段树
如果不知道什么是线段树,请点击这里 线段树
对于这种区间信息,线段树显然是能够维护的。但鉴于本题没有区间修改,线段树显然有点大材小用,并且数组模拟的线段树空间将会达到
线段树还有一些缺点,就是它的查询时间复杂度最坏是的,因为没有区间修改,这个时间开销也略微有点大。
表(倍增)
倍增算法的含义就是成倍增长。我们考虑一个这样的数据结构:
一个二维数组,其中
我们假设一个数据:
可以建立如下所示的表:
建表
我们发现,我们可以一层层地建表。这样,我们就可以通过递推,利用上层的信息,建表。
我们先将读入数据存在这一行。
我们发现,要想实现建立这个表,我们需要每次倍增长度。而最简单的倍增长度方法就是将两段不相交的区间合并起来。
所以我们可以得到如下公式:
这样,我们就能完成建表
for(rg int i=1;i<=16;i++){ // 由计算器可得 log1e5 约为 17,但是这里循环16次已经够了。
for(rg int j=1;j+(1<<i)-1<=n;j++){
st[i][j]=min(st[i-1][j],st[i-1][j+(1<<(i-1))]);
}
}
- 注:为了方便,我们常常把这个表“竖”过来。本篇中的代码一律如此
我们可以发现,建表的时间复杂度是的
查询
我们由上表知道,想要查询的最值,只需求出
其中
比如上方的数据,我们想要查询
我们只需要从往后个, 往前个,肯定能够完全覆盖这个区间
所以我们的查询开销是的
ans=min(st[(int)log2(len)][l],st[(int)log2(len)][r-(1<<((int)log2(len)))+1]);
至此,我们已经完成了全部内容
本文作者:
本文链接:https://www.cnblogs.com/UssEnterprise/p/12081474.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步