题目大意:有一个m*n的方格,每一个格子有他自己的权值。2种操作:
1.改变一个格子的权值。
2.查询所有的x1 <= x <= x2 && y1 <= y <= y2的中,有多少个格子颜色是c。
思路:好像是二维树状数组的样子,但是不知道怎么搞。后来研究了数据范围,发现格子最大300*300,颜色最多才100种,于是算一下300*300*100*4/1024/1024大概是35M,题目要求64M,可以搞了。(这里算的精确一点,我当时没怎么算,吧颜色开成300的了,结果100+M就MLE了。。)
维护100个二维树状数组,每一个代表一种颜色,然后根据容斥原理计算一个方块区间内的数量。更改的时候要改两次,第一次把之前的减去,然后加上改变之后的。
CODE:
#include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #define MAX 310 using namespace std; int m,n; struct Complex{ int tree[MAX][MAX]; void Fix(int x,int y,int c) { for(int i = x;i <= m;i += i&-i) for(int j = y;j <= n;j += j&-j) tree[i][j] += c; } int GetSum(int x,int y) { int re = 0; for(int i = x;i;i -= i&-i) for(int j = y;j;j -= j&-j) re += tree[i][j]; return re; } }tree_arr[110]; int src[MAX][MAX]; int asks; int main() { cin >> m >> n; for(int i = 1;i <= m; ++i) for(int j = 1;j <= n; ++j) { scanf("%d",&src[i][j]); tree_arr[src[i][j]].Fix(i,j,1); } cin >> asks; for(int flag,i = 1;i <= asks; ++i) { scanf("%d",&flag); if(flag == 1) { int x,y,z; scanf("%d%d%d",&x,&y,&z); tree_arr[src[x][y]].Fix(x,y,-1); src[x][y] = z; tree_arr[src[x][y]].Fix(x,y,1); } else { int x1,y1,x2,y2,c; scanf("%d%d%d%d%d",&x1,&x2,&y1,&y2,&c); int ans = tree_arr[c].GetSum(x2,y2); ans -= tree_arr[c].GetSum(x1 - 1,y2); ans -= tree_arr[c].GetSum(x2,y1 - 1); ans += tree_arr[c].GetSum(x1 - 1,y1 - 1); printf("%d\n",ans); } } return 0; }
BZOJ 1452 JSOI 2009 Count 二维树状数组
原文地址:http://blog.csdn.net/jiangyuze831/article/details/39718033