浅谈 倍增/ST表
命题描述
给定一个长度为 \(n\) 的序列,\(m\) 次询问区间最大值
分析
上面的问题肯定可以暴力对吧。
但暴力肯定不是最优对吧,所以我们直接就不考虑了。。。
于是引入:倍增
首先,倍增是个什么东西?
在这里转一篇写的超棒的blog,点我。要是这都没看懂你就连小白兔都不如我就无语了。
总的来说,其实就是倒着运用二分的思想,从需求小的慢慢倍增把答案更新到需求大的
ST表就是一种常见的倍增思想的运用
关于ST表
ST表和树状数组,线段树这两种算法一样,是一种用于解决 \(RMQ(Range Minimum/Maximum Query)\)多次区间查询问题的离线算法
ST表的主要思想是构建一个二维数组 \(st[i][j]\),这个二维数组表示需要查询的数组的从下标 \(i\) 到下标 \(i + 2^{j - 1}\) 的最值,这里以最大值为例。超像 \(dp\) 的说~
我们可以把 \([i, j]\) 拆成数量相等的两半,在求出这两部分的最大值即可,故:
st[i][j] = max(st[i][j - 1], st[i + (1 << (j - 1))][j - 1];
再枚举一下 \(i, j\) 就可以了。点儿都不高级
接下来,该怎么实现查询呢?
如果查询的区间长度刚好是 \(2^k(k为整数)\),直接输出 \(st[l][log(r - l + 1) / log(2)]\) 即可;
如果不是,也很简单,我们还是将所给区间分为两部分。
首先规定在 \(st[i][j]\) 中 \(i = l\),\(j = k\),区间长度 \(len = r - l + 1\)。
会发现最大的\(k\) 应满足 \(2^k <= len\) (这样以 \(l\) 开头的ST表数据覆盖需要查询的区间中的数最多)
所以 \(k = (int) (log(len)/log(2))\)
显然查询时的区间 \([l,r]\) 分成的右边部分的左端点 \(x = r + 1 - 2^p\)
具体实现
#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std;
const int MAXN = 100005;
int st[MAXN][41];
void read(int &a) { // 读优
int k = 1;
a = 0;
char s = getchar();
while(s < '0' || s > '9') {
if(s == '-') k = -1;
s = getchar();
}
while(s >= '0' && s <= '9') {
a = a * 10 + (s - '0');
s = getchar();
}
a *= k;
}
int main() {
int n, m;
read(n); read(m);
for(int i = 1; i <= n; i++) read(st[i][0]);
for(int j = 1; j <= (int)(log(n) / log(2)); j++)
for(int i = 1; i + (1 << j) - 1 <= n; i++)
st[i][j] = max(st[i][j - 1], st[i + (1 << (j - 1))][j - 1]);
for(int i = 1; i <= m; i++) {
int l, r;
read(l); read(r);
int k = (int)(log(r - l + 1) / log(2));
printf("%d\n", max(st[l][k], st[r - (1 << k) + 1][k]));
}
return 0;
}
浅谈 倍增/ST表的更多相关文章
- 浅谈对ST表的一些理解
今天打了人生第一道ST表题(其实只是ST表跑得最快); ST表是一种用来解决RMQ问题的利器... 大体操作有两步: 第一部分nlogn预处理 第二部分O(1)询问 预处理就是运用倍增+区间动规 ST ...
- P7599-[APIO2021]雨林跳跃【二分,倍增,ST表】
正题 题目链接:https://www.luogu.com.cn/problem/P7599 题目大意 \(n\)棵树,在某棵树上时可以选择向左右两边第一棵比它高的树跳,现在\(q\)次询问从\([A ...
- 【BZOJ1047】[HAOI2007]理想的正方形 (倍增ST表)
[HAOI2007]理想的正方形 题目描述 有一个\(a*b\)的整数组成的矩阵,现请你从中找出一个\(n*n\)的正方形区域,使得该区域所有数中的最大值和最小值的差最小. 输入输出格式 输入格式: ...
- 浅谈倍增LCA
题目链接:https://www.luogu.org/problemnew/show/P3379 刚学了LCA,写篇blog加强理解. LCA(Least Common Ancestors),即最近公 ...
- 【编程学习】浅谈哈希表及用C语言构建哈希表!
哈希表:通过key-value而直接进行访问的数据结构,不用经过关键值间的比较,从而省去了大量处理时间. 哈希函数:选择的最主要考虑因素--尽可能避免冲突的出现 构造哈希函数的原则是: ①函数本身便于 ...
- Sql server 浅谈用户定义表类型
1.1 简介 SQL Server 中,用户定义表类型是指用户所定义的表示表结构定义的类型.您可以使用用户定义表类型为存储过程或函数声明表值参数,或者声明您要在批处理中或在存储过程或函数的主体中使用的 ...
- 浅谈MySQL分表
关于分表:顾名思义就是一张数据量很大的表拆分成几个表分别进行存储. 我们先来大概了解以下一个数据库执行SQL的过程: 接收到SQL --> 放入SQL执行队列 --> 使用分析器分解SQL ...
- poj 3264 倍增 ST表
#include<iostream> #include<cmath> using namespace std; ; int a[maxn]; ]; ]; int quick(i ...
- 图书管理(Loj0034)+浅谈哈希表
图书管理 题目描述 图书管理是一件十分繁杂的工作,在一个图书馆中每天都会有许多新书加入.为了更方便的管理图书(以便于帮助想要借书的客人快速查找他们是否有他们所需要的书),我们需要设计一个图书查找系统. ...
随机推荐
- Java9系列第九篇-对HTTP2协议的支持与非阻塞HTTP-API
在HTTP/1.1 发布了16 年之后,IETF在2015年终于通过了HTTP/2 协议.HTTP/2协议旨在降低延迟,满足当今时代对于信息响应时间的要求.在这篇文章中,我会简要的对HTTP/2协议进 ...
- JUC---01阻塞队列(BlockingQueue)
一.什么是阻塞队列 阻塞队列是一个队列,在数据结构中起的作用如上图:当队列是空的,从队列中获取元素的操作将会被阻塞:当队列是满的,从队列中添加元素的操作将会被阻塞 1.为什么需要BlockingQue ...
- Java SFTP 上传、下载等操作
Java SFTP 上传.下载等操作 实际开发中用到了 SFTP 用于交换批量数据文件,然后琢磨了下这方面的东西,基于 JSch 写了个工具类记录下,便于日后使用. JSch是 SSH2 的纯Java ...
- Docker学习—DockerFile
前言: 上一篇文章简单使用了docker 拉取镜像.启动容器.编译镜像:其中编译镜像时,使用到了Dockerfile,那么接下来我们就详细的来说说Dockerfile DockerFile是什么: D ...
- uniapp微信授权操作后事件不触发
微信小程序官网代码 <button form-type="submit" open-type="getUserInfo" bindgetuserinfo= ...
- Docker(3)- Centos 7.x 下 Docker 镜像加速配置
如果你还想从头学起 Docker,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1870863.html 前言 默认情况下,Docker ...
- Blazor 准备好为企业服务了吗?
如果您正在编写 .NET Web 应用程序,您很可能已经意识最近一年在.NET Web开发领域的热点都是 Blazor 的.如果你还没有了解Blazor,它允许您使用 C# 来编写 Web UIs,传 ...
- 无法将“add-migration”项识别为 cmdlet、函数、脚本文件或可运行程序的名称。解决方案
在程序包管理控制台中执行 Install-Package Microsoft.EntityFrameworkCore.Tools
- 中介者模式及在NetCore中的使用MediatR来实现
在现实生活中,常常会出现好多对象之间存在复杂的交互关系,这种交互关系常常是"网状结构",它要求每个对象都必须知道它需要交互的对象.例如,每个人必须记住他(她)所有朋友的电话:而且, ...
- Javascript基础知识面试题
[]==![] 解题思路: step1, []==!Boolean([]);step2, []==!true;step3, []==false;step4, [].valueOf()= ...