【算法模板】数据结构:ST表

【算法模板】数据结构:ST表

概念

ST 表是一种基于 倍增 思想,用于解决 可重复贡献问题 的数据结构

可重复贡献问题 是指对于运算 opt⁡opt,满足 𝑥opt⁡𝑥=𝑥x optx =x ,则对应的区间询问就是一个可重复贡献问题。例如,最大值有 max⁡(𝑥,𝑥)=𝑥max(x ,x )=x ,gcd 有 gcd⁡(𝑥,𝑥)=𝑥gcd(x ,x )=x,所以 RMQ 和区间 GCD 就是一个可重复贡献问题。像区间和就不具有这个性质,如果求区间和的时候采用的预处理区间重叠了,则会导致重叠部分被计算两次,这是我们所不愿意看到的。另外,opt⁡opt 还必须满足结合律才能使用 ST 表求解。

ST 表基于倍增 思想,可以做到 𝑂(n logn) 预处理,O*(1) 回答每个询问。但是不支持修改操作。

所谓预处理,即我们需要预先处理出,对于每个位置 i,我们需要知道,以 i 作为左端点, i+2^j-1作为右端点的每个区间的最大值。

由于 log2n 较小,所以我们可以直接枚举每个 j。

我们用一个数组表示,f[i][j] 表示以 i 为左端点,区间长度为2^j ,也就是右端点是 i+2^j-1。

模板

cpp 复制代码
vector<vector<int>> ST_prework(const vector<int> &v) {
    int n = v.size();
    int t = log2(n)+1;
    vector<vector<int>> ST(n, vector<int>(t));
    for (int i = 0; i < n; i++)ST[i][0] = v[i];
    for (int j = 1; j < t; j++)//枚举区间长度
        for (int i = 0; i + (1 << j) - 1 < n; i++)
            ST[i][j] = max(ST[i][j - 1], ST[i + (1 << (j - 1))][j - 1]);
    return ST;
}
int ST_query(vector<vector<int>> &ST,int l, int r) {
    int k = log2(r - l + 1);//左闭右闭
    return max(ST[l][k], ST[r - (1 << k) + 1][k]);
}

例题

区间最大值

区间最大值 - 蓝桥云课 (lanqiao.cn)

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
int main() {
    int n,q;cin>>n>>q;
    const int t=log2(n)+1;
    vector<vector<int>> st(n,vector<int>(t));
    for(int i=0;i<n;i++)cin>>st[i][0];
    for(int i=1;i<t;i++)
        for(int j=0;j+(1<<(i-1))<n;j++)
            st[j][i]=max(st[j][i-1],st[j+(1<<(i-1))][i-1]);
    while(q--){
        int l,r;cin>>l>>r;
        l--,r--;
        int x=log2(r-l+1);
        cout<<max(st[l][x],st[r-(1<<x)+1][x])<<endl;
    }
    return 0;
}

Balanced Lineup G

USACO07JAN Balanced Lineup G - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
int main() {
    int n, q;
    cin >> n >> q;
    const int t = log2(n) + 1;
    vector<vector<int>> stmax(n, vector<int>(t));
    for (int i = 0; i < n; i++)cin >> stmax[i][0];
    auto stmin = stmax;
    for (int j = 1; j < t; j++) {
        for (int i = 0; i + (1 << (j - 1)) < n; i++) {
            stmax[i][j] = max(stmax[i][j - 1], stmax[i + (1 << (j - 1))][j - 1]);
            stmin[i][j] = min(stmin[i][j - 1], stmin[i + (1 << (j - 1))][j - 1]);
        }
    }
    while(q--){
        int a,b;cin>>a>>b;
        --a,--b;
        int x=log2(b-a+1);
        int ans=0;
        ans+=max(stmax[a][x],stmax[b-(1<<x)+1][x]);
        ans-=min(stmin[a][x],stmin[b-(1<<x)+1][x]);
        cout<<ans<<endl;
    }
    return 0;
}
相关推荐
Not Dr.Wang42227 分钟前
自动控制系统稳定性研究及判据分析
算法
VT.馒头28 分钟前
【力扣】2722. 根据 ID 合并两个数组
javascript·算法·leetcode·职场和发展·typescript
EnglishJun28 分钟前
数据结构的学习(四)---栈和队列
数据结构·学习
ffqws_33 分钟前
A*算法:P5507 机关 题解
算法
执着2591 小时前
力扣hot100 - 108、将有序数组转换为二叉搜索树
算法·leetcode·职场和发展
2501_901147831 小时前
学习笔记:单调递增数字求解的迭代优化与工程实践
linux·服务器·笔记·学习·算法
数智工坊1 小时前
【数据结构-特殊矩阵】3.5 特殊矩阵-压缩存储
数据结构·线性代数·矩阵
AI科技星1 小时前
张祥前统一场论核心场方程的经典验证-基于电子与质子的求导溯源及力的精确计算
线性代数·算法·机器学习·矩阵·概率论
kebijuelun1 小时前
ERNIE 5.0:统一自回归多模态与弹性训练
人工智能·算法·语言模型·transformer
芝士爱知识a1 小时前
AlphaGBM 深度解析:下一代基于 AI 与蒙特卡洛的智能期权分析平台
数据结构·人工智能·python·股票·alphagbm·ai 驱动的智能期权分析·期权