二维ST表

二维ST表。

​ 二维ST表咋写捏?

(f[x][y][k][l])表示左上角为((x, y)),右下角为((x + 2^{k} -1 , y+2^l - 1))的矩形内的最值是多少。

​ 转移的话和原来差不多,首先可以将矩形从中间横着劈一刀:

(f[x][y][k][l] = max/min(f[x][y][k - 1][l], f[x + 2 ^ {k - 1}][y][k - 1][l]))

​ 然后还可以将矩形从中间竖着劈一刀:

(f[x][y][k][l] = max/min(f[x][y][k][l - 1], f[x][y + 2^{l - 1}][k][l - 1]))

​ 同样的,还是可以实现(O(1))查询,假设现在要查询左上角为((x1, y1)),右下角为((x2 , y2))的矩形内的最值是多少。

​ 设(p = lg[x2 - x1 + 1], q = lg[y2 - y1 + 1])

​ 如图我们可以把查询的大矩形分为四个小矩形,他们的坐标分别是:
((x1, y1, x1 + 2^p - 1, y1 + 2^q - 1)),
((x1, y2 - 2 ^ q + 1, x2 - 2^p + 1, y2)),
((x2 - 2 ^ p + 1, y1, x2, y2 - 2^q + 1)),
((x2 - 2^p + 1, y2 - 2^k + 1, x2, y2))
分别对应图上的(w1,w2,w3,w4)。对这四个小矩形的最值取(max/min),就是大矩形的最值了。

P2216 [HAOI2007]理想的正方形

​ 这道题可用二维ST表做。

题目链接

​ 对于这道题直接开(f[1005][1005][11][11])是会爆空间的,但是我们发现题目要求的是正方形,其实就可以省去第三,四维;直接维护每个正方形的值就好了。

#include <bits/stdc++.h>

using namespace std;

inline long long read() {
    long long s = 0, f = 1; char ch;
    while(!isdigit(ch = getchar())) (ch == '-') && (f = -f);
    for(s = ch ^ 48;isdigit(ch = getchar()); s = (s << 1) + (s << 3) + (ch ^ 48));
    return s * f;
}

const int N = 1005, inf = 2e9;
int n, m, k, ans;
int lg[N], f[N][N], g[N][N];

int main() {

    n = read(); m = read(); k = read();
    for(int i = 1;i <= n; i++)
        for(int j = 1;j <= m; j++) f[i][j] = g[i][j] = read();

    
    lg[0] = -1;
    for(int i = 1;i <= k; i++) lg[i] = lg[i >> 1] + 1;

    for(int s = 1;s <= lg[k]; s++) 
        for(int i = 1;i + (1 << s) - 1 <= n; i++)
            for(int j = 1;j + (1 << s) - 1 <= m; j++) {
                f[i][j] = max(max(f[i][j], f[i][j + (1 << (s - 1))]), max(f[i + (1 << (s - 1))][j], f[i + (1 << (s - 1))][j + (1 << (s - 1))]));
                g[i][j] = min(min(g[i][j], g[i][j + (1 << (s - 1))]), min(g[i + (1 << (s - 1))][j], g[i + (1 << (s - 1))][j + (1 << (s - 1))]));
            }

    ans = inf;
    for(int i = 1;i + k - 1 <= n; i++)
        for(int j = 1;j + k - 1 <= m; j++) {
            int x = i + k - 1, y = j + k - 1; 
            int maxn = max(max(f[i][j], f[x - (1 << lg[k]) + 1][j]), max(f[i][y - (1 << lg[k]) + 1], f[x - (1 << lg[k]) + 1][y - (1 << lg[k]) + 1]));
            int minn = min(min(g[i][j], g[x - (1 << lg[k]) + 1][j]), min(g[i][y - (1 << lg[k]) + 1], g[x - (1 << lg[k]) + 1][y - (1 << lg[k]) + 1]));
            ans = min(ans, maxn - minn);
        }
    
    printf("%d
", ans);

    return 0;
}
原文地址:https://www.cnblogs.com/czhui666/p/13693835.html