[蓝桥杯学习] ST表

RMQ问题

ST 表

用状态 s[i][j] 记录区间长度为 2^j 的长度的区间的最大值

所以状态转移方程就是 st[i][j] = max( st[i][j-1] , st[i+(1 << (j-1))][j-1] )

注意状态转移的方向,保证区间合法性(i+2^j 不能超过数组大小)

写完这些后,定义好第一个,就可以从前往后进行计算

用ST表进行区间查询

ST表存储的区间是2的整数倍,所以要计算的是,如何从要求的区间,到ST表存储的区域。

要寻找一个k,如果满足以下的大小关系,

就可以取两个区间的最大值 max(st[l][k],st[r-(1<<k)][k]),这两个区间是囊括了整个要求的区域。

k值的具体计算是,把(r-l+1)对2求对数,并向下取整,可以用强制类型转换来实现。

求区间最大值的代码

cpp 复制代码
#include <iostream>
#include <algorithm>
#include <cmath>
using namespace std;

#define ll long long

const int N = 5e5 + 5;
int n,q;
ll a[N];
ll st[N][21];

ll getMax(int l,int r)
{
  //计算k,区间长度对2取对数
  int k = log(r-l+1)/log(2);
  return max(st[l][k],st[r-(1<<k)+1][k]); 
}


int main()
{
  // 请在此输入您的代码
  cin >> n >> q;
  for(int i = 1 ; i <= n ; i++)
  {
    int x;
    cin >> x;
    a[i] = x;
  }
  //构造ST表
  //1.初始化
  for(int i = 1 ; i <= n ; i++) st[i][0] = a[i];
  //2.利用状态转移方程求ST表
  for(int j = 1 ; j <= 20 ; j++)
  {
    for(int i = 1 ; i <= n ; i++)
    {
      if(i + (1<<j) -1 <= n)   //不要忘记-1,是要区间长度为 2^j 的
      {
        st[i][j] = max(st[i][j-1],st[i+(1<<(j-1))][j-1]);
      }
    }
  }
  //3.利用ST表来求区间最大值
  while(q--)
  {
    int l,r;
    cin >> l >> r;
    cout << getMax(l,r) <<'\n';
  }



  return 0;
}
相关推荐
做cv的小昊1 小时前
计算机图形学:【Games101】学习笔记05——着色(插值、高级纹理映射)与几何(基本表示方法)
笔记·opencv·学习·计算机视觉·图形渲染·几何学
车载测试工程师1 小时前
CAPL学习-CAN相关函数-统计API函数
网络·网络协议·学习·capl·canoe
好奇龙猫2 小时前
【AI学习-comfyUI学习-第二十四节-open(contorlnet多重处理)+图生图openpose-各个部分学习】
人工智能·学习
wanzhong23333 小时前
CUDA学习5-矩阵乘法(共享内存版)
深度学习·学习·算法·cuda·高性能计算
PNP Robotics5 小时前
PNP机器人受邀参加英业达具身智能活动
大数据·人工智能·python·学习·机器人
iconball5 小时前
个人用云计算学习笔记 --24 虚拟化、KVM 基础使用与热迁移实验、VMware ESXi笔记
运维·笔记·学习·云计算
奕辰杰7 小时前
Netty私人学习笔记
笔记·学习·netty·网络通信·nio
Larry_Yanan8 小时前
Qt多进程(五)QUdpSocket
开发语言·c++·qt·学习·ui
De-Alf8 小时前
Megatron-LM学习笔记(6)Megatron Model Attention注意力与MLA
笔记·学习·算法·ai
Gary Studio9 小时前
MPP充电学习笔记
笔记·学习