Uva 11297 Census,二维线段树,板子

维护二维矩阵。

q  x1, y1, x2, y2 :   查询x1<=x<=x2, y1<=y<=y2的格子(x,y)的最大值和最小值。

c    x, y, v:    修改(x,y) = v。

矩阵大小(n,m<=500)。

查询次数q<=40000。

#include<algorithm>
using namespace std;

const int INF = 1<<30;
const int maxn = 2000 + 10;

int A[maxn][maxn];

struct IntervalTree2D {
    int Max[maxn][maxn], Min[maxn][maxn], n, m;
    int xo, xleaf, row, x1, y1, x2, y2, x, y, v, vmax, vmin; // 参数、查询结果和中间变量

    void query1D(int o, int L, int R) {
        if(y1 <= L && R <= y2) {
            vmax = max(Max[xo][o], vmax);
            vmin = min(Min[xo][o], vmin);
        } else {
            int M = L + (R-L)/2;
            if(y1 <= M) query1D(o*2, L, M);
            if(M < y2) query1D(o*2+1, M+1, R);
        }
    }

    void query2D(int o, int L, int R) {
        if(x1 <= L && R <= x2) {
            xo = o;
            query1D(1, 1, m);
        } else {
            int M = L + (R-L)/2;
            if(x1 <= M) query2D(o*2, L, M);
            if(M < x2) query2D(o*2+1, M+1, R);
        }
    }

    void modify1D(int o, int L, int R) {
        if(L == R) {
            if(xleaf) {
                Max[xo][o] = Min[xo][o] = v;
                return;
            }
            Max[xo][o] = max(Max[xo*2][o], Max[xo*2+1][o]);
            Min[xo][o] = min(Min[xo*2][o], Min[xo*2+1][o]);
        } else {
            int M = L + (R-L)/2;
            if(y <= M) modify1D(o*2, L, M);
            else modify1D(o*2+1, M+1, R);
            Max[xo][o] = max(Max[xo][o*2], Max[xo][o*2+1]);
            Min[xo][o] = min(Min[xo][o*2], Min[xo][o*2+1]);
        }
    }

    void modify2D(int o, int L, int R) {
        if(L == R) {
            xo = o;
            xleaf = 1;
            modify1D(1, 1, m);
        } else {
            int M = L + (R-L)/2;
            if(x <= M) modify2D(o*2, L, M);
            else modify2D(o*2+1, M+1, R);
            xo = o;
            xleaf = 0;
            modify1D(1, 1, m);
        }
    }

    // 只构建xo为叶子(即x1=x2)的y树
    void build1D(int o, int L, int R) {
        if(L == R) Max[xo][o] = Min[xo][o] = A[row][L];
        else {
            int M = L + (R-L)/2;
            build1D(o*2, L, M);
            build1D(o*2+1, M+1, R);
            Max[xo][o] = max(Max[xo][o*2], Max[xo][o*2+1]);
            Min[xo][o] = min(Min[xo][o*2], Min[xo][o*2+1]);
        }
    }

    void build2D(int o, int L, int R) {
        if(L == R) {
            xo = o;
            row = L;
            build1D(1, 1, m);
        } else {
            int M = L + (R-L)/2;
            build2D(o*2, L, M);
            build2D(o*2+1, M+1, R);
            for(int i = 1; i <= m*4; i++) {
                Max[o][i] = max(Max[o*2][i], Max[o*2+1][i]);
                Min[o][i] = min(Min[o*2][i], Min[o*2+1][i]);
            }
        }
    }

    void query() {
        vmax = -INF;
        vmin = INF;
        query2D(1, 1, n);
    }

    void modify() {
        modify2D(1, 1, n);
    }

    void build() {
        build2D(1, 1, n);
    }
};

IntervalTree2D t;

#include<cstdio>

int main() {
    int n, m, Q, x1, y1, x2, y2, x, y, v;
    char op[10];
    scanf("%d%d", &n, &m);
    t.n = n;
    t.m = m;
    for(int i=1; i<=n; ++i)
        for(int j=1; j<=m; ++j)
            scanf("%d", &A[i][j]);
    t.build();

    scanf("%d", &Q);
    while(Q--) {
        scanf("%s", op);
        if(op[0]=='q') {
            scanf("%d%d%d%d", &t.x1, &t.y1, &t.x2, &t.y2);
            t.query();
            printf("%d %d\n", t.vmax, t.vmin);
        } else {
            scanf("%d%d%d", &t.x, &t.y, &t.v);
            t.modify();
        }
    }
    return 0;
}

Uva 11297 Census,二维线段树,板子

时间: 2024-12-27 14:50:54

Uva 11297 Census,二维线段树,板子的相关文章

UVA 11297 Census ——二维线段树

[题目分析] 二维线段树模板题目. 简直就是无比的暴力.时间复杂度为两个log. 标记的更新方式比较奇特,空间复杂度为N^2. 模板题目. [代码] #include <cstdio> #include <cstring> //#include <cmath> #include <cstdlib> #include <map> #include <set> #include <queue> #include <str

Uva 11297.Census——二维线段树

http://www.bnuoj.com/v3/problem_show.php?pid=19715 二维线段树模板 #include <cstring> #include <cstdio> #include <algorithm> #include <iostream> #define lson l,mid,rt<<1 #define rson mid+1,r,rt<<1|1 #define rep(i,n) for(int i=0

Uva 11297 Census 二维线段树

题目链接:点击打开链接 好久没发题解了, 第一维的线段树更新到底,叶子节点建一棵线段树. #include<iostream> #include<cstring> #include<cstdio> #include<cmath> #include<map> #include<vector> #include<set> #include<string> #include<algorithm> usin

hdu 1823 Luck and Love ,二维线段树

Luck and Love Time Limit: 10000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 5282    Accepted Submission(s): 1324 Input 本题有多个测试数据,第一个数字M,表示接下来有连续的M个操作,当M=0时处理中止. 接下来是一个操作符C. 当操作符为'I'时,表示有一个MM报名,后面接着一个整数,H表示身

HDU1832 二维线段树求最值(模板)

Luck and Love Time Limit: 10000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 50 Accepted Submission(s): 20   Problem Description 世界上上最远的距离不是相隔天涯海角而是我在你面前可你却不知道我爱你                ―― 张小娴 前段日子,枫冰叶子给Wiskey做了个征婚启事,聘

ZOJ 2859 二维线段树

思路:自己写的第二发二维线段树1A,哈哈,看来对二维的push操作比较了解了:但是还没遇到在两个线段树中同时进行push操作的,其实这题我是想在x维和y维同时进行push操作的,但是想了好久不会,然后看到这题又给出10秒,然后想想在x维线段直接单点查询肯定也过了,然后在第二维就只有pushup操作,在第一维线段树没有pushup操作.要是在第一维也有pushup操作的话,那就不用单点查询那么慢了.不过也A了,想找题即在二维同时进行pushup和pushdown操作的. #include<iost

[POJ2155] Matrix(二维线段树,树套树)

题目链接:http://poj.org/problem?id=2155 题意:给一个01矩阵,两个操作,翻转:子矩阵里每一个数都由0变1,1变0. 查询:查询某一点是0还是1. 一直以为二维线段树就是开一个线段树数组的我- 这题暴力更新每一个小矩形,翻转就+1,最后看看某点的奇偶. 写屎了,特别注意的是在外层查询的时候要先把当前层的内层query掉,接着再向下扩展.这样外层就不用lazy啦 1 #include <algorithm> 2 #include <iostream> 3

poj1195 Mobile phones 二维线段树入门

二维线段树就是树套树,线段树套线段树... #include<iostream> #include<cstdio> #include<cstring> #include<cstdlib> #include<algorithm> #define REP(i,a,b) for(int i=a;i<=b;i++) #define MS0(a) memset(a,0,sizeof(a)) #define lson l,m,rt<<1 #

tyvj P1716 - 上帝造题的七分钟 二维树状数组区间查询及修改 二维线段树

P1716 - 上帝造题的七分钟 From Riatre    Normal (OI)总时限:50s    内存限制:128MB    代码长度限制:64KB 背景 Background 裸体就意味着身体. 描述 Description “第一分钟,X说,要有矩阵,于是便有了一个里面写满了0的n×m矩阵.第二分钟,L说,要能修改,于是便有了将左上角为(a,b),右下角为(c,d)的一个矩形区域内的全部数字加上一个值的操作.第三分钟,k说,要能查询,于是便有了求给定矩形区域内的全部数字和的操作.第