zoukankan      html  css  js  c++  java
  • ST表--适用多查询O(1)

    ST表类似树状数组,线段树这两种算法,是一种用于解决RMQ问题的离线算法。

    预处理(O(nlogn)),查询(O(1))

    给出一个长度为(N)的数列,和(M)次查询,求出每一次询问的区间内数字的最大值

    定义:

    (st[i][j])表示从(i)位置开始的(2^j)个数中的最大值,即(st[i][j] = max(a_i,...a_{i + 2^j-1}))

    预处理

    (O(nlogn))

    img

    (st[i][j])对应的([i,i + 2^j - 1])分成([i,i + 2^{j - 1} - 1])([i+2^{j - 1},i+2^j-1])两部分

    对应的是(st[i][j - 1])(st[i + (1 << j) - 1][j - 1])

    [st[i][j] = max(st[i][j - 1], st[i + (1 << (j - 1))][j - 1]); ]

    查询

    (O(1))

    img

    ([l,r])分成两个有交集的区间([l,l + 2^k-1])([r - 2^k + 1][r])

    [k = log2(r - l + 1)\ ans = max(st[l][k],st[r - (1 << k) + 1][k]) ]

    打表观察

    #include <iostream>
    #include <cstdio>
    #include <cmath>
    using namespace std;
    const int N = 3e5 + 5;
    int st[N][22];
    int main(){
        int n = 20;
        for(int j = 1; j <= 20; j++)
            for(int i = 1; i + (1 << j) - 1 <= n; i++)
                printf("%d [%d,%d] [%d,%d]
    ", 1 << (j - 1), i, i + (1 << (j - 1)) - 1, i + (1 << (j - 1)), i + (1 << j) - 1);
        return 0;
    }
    

    {{uploading-image-762726.png(uploading...)}}

    可以观察其变化情况

    模板

    #include <iostream>
    #include <cstdio>
    #include <cmath>
    using namespace std;
    const int N = 1e5 + 5;
    int st[N][20];
    int query(int l, int r){
        int k = log2(r - l + 1);
        return max(st[l][k], st[r - (1 << k) + 1][k]);
    }
    int main(){
        int n, m;
        scanf("%d%d", &n, &m);
        for(int i = 1; i <= n; i++)
            scanf("%d", &st[i][0]);
        
      int k = log2(n / 2) + 1; //常数优化
        for(int j = 1; j <= 17; j++) // j in [1, k]
            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;
            scanf("%d%d", &l, &r);
            printf("%d
    ", query(l, r));
        }
        return 0;
    }
    

    扩展

    st表必定是一个单调函数

    • 维护最大值的st表 - 单调递增
    • 维护最小值的st表 - 单调递减

    可以用二分去寻找满足条件的区间个数

    遍历下左区间,然后二分在st表里查找右区间

    同时,st表也可以去维护其他的,比如区间LCA

    总之,对于区间RMQ问题,或者区间问题,多组查询,且不修改,ST表很好用

  • 相关阅读:
    修改nuget包默认存放路径,避免增加C盘的负担
    .Net Core 3.0 (一):安装Vs2019
    .NET Core 学习资料
    SQLSERVER查询整个数据库中某个特定值所在的表和字段的方法
    MySql 时间查询
    如何设置IIS程序池的回收时间,才能最大程度的减少对用户的影响?
    SqlServer 获取工作日(周六、周日休息,周六日不休息,周六不休息)
    SQL Server 删除数据库中表数据
    SQL Server 删除数据库所有表和所有存储过程
    摘要
  • 原文地址:https://www.cnblogs.com/Emcikem/p/11561412.html
Copyright © 2011-2022 走看看