【模板】树状数组
题目描述
如题,已知一个数列,你需要进行下面两种操作:
-
将某一个数加上 x x x
-
求出某区间每一个数的和
输入格式
第一行包含两个正整数 n , m n,m n,m,分别表示该数列数字的个数和操作的总个数。
第二行包含 n n n 个用空格分隔的整数,其中第 i i i 个数字表示数列第 i i i 项的初始值。
接下来 m m m 行每行包含 3 3 3 个整数,表示一个操作,具体如下:
-
1 x k
含义:将第 x x x 个数加上 k k k -
2 x y
含义:输出区间 [ x , y ] [x,y] [x,y] 内每个数的和
输出格式
输出包含若干行整数,即为所有操作 2 2 2 的结果。
样例 #1
样例输入 #1
5 5
1 5 4 2 3
1 1 3
2 2 5
1 3 -1
1 4 2
2 1 4
样例输出 #1
14
16
提示
【数据范围】
对于
30
%
30\%
30% 的数据,
1
≤
n
≤
8
1 \le n \le 8
1≤n≤8,
1
≤
m
≤
10
1\le m \le 10
1≤m≤10;
对于
70
%
70\%
70% 的数据,
1
≤
n
,
m
≤
1
0
4
1\le n,m \le 10^4
1≤n,m≤104;
对于
100
%
100\%
100% 的数据,
1
≤
n
,
m
≤
5
×
1
0
5
1\le n,m \le 5\times 10^5
1≤n,m≤5×105。
样例说明:
故输出结果14、16
思路
顾名思义树状数组模板题,打个模板即可。
上代码
#include<bits/stdc++.h>
using namespace std;
const int N = 500010;
int n,m,a[N],c[N];
void add(int k, int i){
while(i<=n){
c[i]+=k;
i+=(i&-i);
}
}
int sum(int num){
int cnt = 0;
while(num){
cnt+=c[num];
num-=(num&-num);
}
return cnt;
}
int main(){
cin >> n >> m;
for(int i=1;i<=n;i++){
cin >> a[i];
add(a[i],i);
}
int f,x,y;
while(m--){
scanf("%d%d%d",&f,&x,&y);
if(f==1){
add(y,x);
}else{
cout << sum(y)-sum(x-1) << endl;
}
}
return 0;
}