【题解】[JSOI2009]计数问题

Problem

\(\text{Solution:}\)

开始有一种暴力的做法:对每一行维护 \(100\) 个树状数组对应 \(100\) 个颜色。查询枚举行来查询。

复杂度:\(O(m\cdot \log n\cdot q)\) 过不去的样子。

考虑用二维树状数组,直接维护二维矩阵。修改与查询的复杂度都做到了 \(O(\log n\cdot \log m).\)

于是最终复杂度可以做到 \(O(q\log n\log m).\)

注意树状数组实现上的细节:第二重循环要保证每次都是从 \(y\) 开始,所以要重新复制一个变量来做。而且两个维度的最大值也是不一样的,上界不同修改的时候要注意。

这题的树状数组显然没有 上帝造题的七分钟 那个树状数组难。这题算是入门版的二维树状数组。

维护矩阵同时维护一个颜色即可

#include<bits/stdc++.h>
using namespace std;
int tr[301][301][101],n,m,a[301][301],q;
inline int lowbit(int x){return x&(-x);}
void change(int x,int y,int c,int v){
	for(;x<=n;x+=lowbit(x))
		for(int i=y;i<=m;i+=lowbit(i))
			tr[x][i][c]+=v;
}
int query(int x,int y,int c){
	int res=0;
	for(;x;x-=lowbit(x))
		for(int i=y;i;i-=lowbit(i))
			res+=tr[x][i][c];
	return res;
}
int main(){
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;++i)
		for(int j=1;j<=m;++j){
			scanf("%d",&a[i][j]);
			change(i,j,a[i][j],1);
		}
	scanf("%d",&q);
	for(;q;q--){
		int opt;
		scanf("%d",&opt);
		if(opt==1){
			int x,y,c;
			scanf("%d%d%d",&x,&y,&c);
			change(x,y,a[x][y],-1);
			a[x][y]=c;
			change(x,y,a[x][y],1);
		}
		else{
			int ax,bx,ay,by,c;
			scanf("%d%d%d%d%d",&ax,&bx,&ay,&by,&c);
			int ans=query(bx,by,c)-query(ax-1,by,c)-query(bx,ay-1,c)+query(ax-1,ay-1,c);
			printf("%d\n",ans);
		}
	}
	return 0;
}
posted @ 2021-06-28 15:09  Refined_heart  阅读(49)  评论(0编辑  收藏  举报