ST表的简单应用

本文介绍了ST表,一种用于解决区间最值问题的数据结构,通过预处理能在O(nlogn)时间内处理数据。文章详细阐述了ST表的原理,如何使用状态转移方程进行区间查询,并给出了一个实际问题(洛谷P1816忠诚)的示例。
摘要由CSDN通过智能技术生成

1. ST表简介

ST表是基于倍增思想,用于解决重复贡献问题的数据结构。
ST表最广泛的应用就是解决RMQ问题(区间最值问题),ST表可以通过O(n logn)的时间进行预处理,O(1)的时间进行问题查询。但是ST表不支持单点修改和区间修改

2.ST表的原理

ST表利用了重复贡献问题,例如: M a x ( n , n ) = n , M i n ( n , n ) = n , G C D ( n , n ) = n Max(n, n) = n, Min(n, n) = n,GCD(n, n) = n Max(n,n)=n,Min(n,n)=nGCD(n,n)=n ,求解时区间有重叠部分并不会影响我们对结果的计算,所以 RMQ 和区间 GCD 就是一个可重复贡献问题。像区间和就不具有这个性质,如果求区间和的时候采用的预处理区间重叠了,则会导致重叠部分被计算两次,这是我们所不愿意看到的。

我们用 f [ i ] [ j ] f[i][j] f[i][j]来表示从第 i i i 个数起连续 2 j 2^j 2j 个数中的最小值,也就是区间 [ i , i + 2 j − 1 ] [i,i+2^j- 1] [i,i+2j1]的最值。

显然 f [ i ] [ 0 ] = x i , f[i][0] = x_i, f[i][0]=xi我们可以得到状态转移方程:

f [ i ] [ j ] = m a x ( f [ i ] [ j − 1 ] , f [ i + 2 j − 1 ] [ j − 1 ] ) ; f[i][j] = max(f [i][ j-1],f [i + 2^{j-1}][ j - 1]); f[i][j]=max(f[i][j1]f[i+2j1][j1])

所以在预处理的时候,我们不妨先枚举区间长度 j j j,再枚举区间左端点 i i i,只要能保证 i + ( 1 < < j ) − 1 ≤ n i+(1<<j)-1≤n i+(1<<j)1n,那么我们便可以将各种区间给预处理出来。

看不懂的可以自己跟着代码试着走一下

void st(int n)
{
    for (int j = 1; j <= 20; j++) //从j开始枚举区间长度
        for (int i = 1; i <= n; i++)
            if (i + (1 << j)-1 <= n)
                f[i][j] = min(f[i][j - 1], f[i + (1 << (j - 1))][j - 1]);
}

以上就是预处理部分。而对于查询,可以简单实现 如下:

我们可以找距离区间长度 ( R − L + 1 ) (R-L+1) RL+1最近的一个 2 n 2^n 2n 次方数 k k k ,那么我们就可以用 L L L 为左端点 长度为 k k k的区间 a n d and and R R R 为右端点 区间长度为 k k k 的两个区间 来更新所求的区间。

不懂的可以画画图!

也就是说对于每个询问区间 [ l , r ] [l,r] [lr],我们把它分成两个部分: [ l , l + 2 s − 1 ] [l,l+2^s-1] [l,l+2s1] [ r − 2 s + 1 , r ] [r-2^s+1,r] [r2s+1r],其中, s = ⌊ l o g 2 ( r − l + 1 ) ⌋ s = ⌊log_2(r-l+1)⌋ s=log2(rl+1)⌋(下取整),两部分结果的最大值就是回答。

请添加图片描述

3.例题

洛谷 P1816 忠诚

题目描述

老管家是一个聪明能干的人。他为财主工作了整整 10 10 10 年。财主为了让自已账目更加清楚,要求管家每天记 k k k 次账。由于管家聪明能干,因而管家总是让财主十分满意。但是由于一些人的挑拨,财主还是对管家产生了怀疑。于是他决定用一种特别的方法来判断管家的忠诚,他把每次的账目按 1 , 2 , 3 … 1, 2, 3 \ldots 1,2,3 编号,然后不定时的问管家问题,问题是这样的:在 a a a b b b 号账中最少的一笔是多少?为了让管家没时间作假,他总是一次问多个问题。

输入格式

输入中第一行有两个数 m , n m, n m,n,表示有 m m m 笔账 n n n 表示有 n n n 个问题。

第二行为 m m m 个数,分别是账目的钱数。

后面 n n n 行分别是 n n n 个问题,每行有 2 2 2 个数字说明开始结束的账目编号。

输出格式

在一行中输出每个问题的答案,以一个空格分割。

样例 #1

样例输入 #1

10 3
1 2 3 4 5 6 7 8 9 10
2 7
3 9
1 10

样例输出 #1

2 3 1

蒟蒻代码

#include <bits/stdc++.h>
using namespace std;

#define int long long
#define xx first
#define yy second
typedef pair<int, int> PII;
const int inf = 0x3f3f3f3f;
const int N = 1e5 + 10;

int v[N], n, m, k, x, t, T;
int f[N][20];

void ClearFloat()
{
    ios_base::sync_with_stdio(false);
    cin.tie(0);
    cout.tie(0);
}

void st(int n)
{
    for (int j = 1; j <= 20; j++)
        for (int i = 1; i <= n; i++)
            if (i + (1 << j)-1 <= n)
                f[i][j] = min(f[i][j - 1], f[i + (1 << (j - 1))][j - 1]);
}

signed main()
{
    ClearFloat();
    cin >> n >> m;
    for (int i = 1; i <= n; i++)
        cin >> f[i][0];
    st(n);
    while (m--)
    {
        int l, r;
        cin >> l >> r;

        int k = log2(r - l + 1);
        cout << min(f[l][k], f[r - (1 << k) + 1][k]) << " ";
    }
    
    return 0;
}
记录一下第一次用MarkDown语法写的文章 😁
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值