#133. 二维树状数组 1:单点修改,区间查询

题目描述

这是一道模板题。

给出一个 n\times mn×m 的零矩阵 AA,你需要完成如下操作:

  • 1 x y k:表示元素 A_{x,y}Ax,y 自增 kk;
  • 2 a b c d:表示询问左上角为 (a,b)(a,b),右下角为 (c,d)(c,d) 的子矩阵内所有数的和。
输入格式

输入的第一行有两个正整数 n,mn,m;
接下来若干行,每行一个操作,直到文件结束。

输出格式

对于每个 2 操作,输出一个整数,表示对于这个操作的回答。

样例
样例输入
2 2
1 1 1 3
1 2 2 4
2 1 1 2 2
样例输出
7
数据范围与提示

对于 10\%10% 的数据,n=1n=1;
对于另 10\%10% 的数据,m=1m=1;
对于全部数据,1\le n,m\le 2^{12},1\le x,a,c\le n,1\le y,b,d\le m,|k|\le 10^51n,m212,1x,a,cn,1y,b,dm,k105,保证操作数目不超过 3\times 10^53×105,且询问的子矩阵存在。

 
#include<bits/stdc++.h>

using namespace std;
int n,m;
const int N=5000;
long long s[N][N];

int lowbit(int x)
{
    return x&(-x);
}

void updata(int x,int y,int z)
{
    for(int i=x;i<=n;i+=lowbit(i)){
        for(int j=y;j<=m;j+=lowbit(j)){
            s[i][j]+=z;
        }
    }
}

long long sum(int x,int y)
{
    long long res=0;
    for(int i=x;i>0;i-=lowbit(i)){
        for(int j=y;j>0;j-=lowbit(j)){
            res+=s[i][j];
        }
    }
    return res;
}
int main()
{
    memset(s,0,sizeof(s));
    scanf("%d %d",&n,&m);
    int k;
    while(scanf("%d",&k)==1){
        if(k==1){
            int x,y,z;
            scanf("%d %d %d",&x,&y,&z);
            updata(x,y,z);
        }
        else{
            int x1,y1,x2,y2;
            scanf("%d %d %d %d",&x1,&y1,&x2,&y2);
            printf("%lld\n",sum(x2,y2)+sum(x1-1,y1-1)-sum(x1-1,y2)-sum(x2,y1-1));
        }
    }

    return 0;
}

 

 

转载于:https://www.cnblogs.com/chenchen-12/p/10188331.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值