PS:如果读过题了可以跳过题目描述直接到题解部分
提交链接:洛谷 P3374 【模板】树状数组 1
题目
题目描述
如题,已知一个数列,你需要进行下面两种操作:
-
将某一个数加上 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
代码实现
100pts
//洛谷 P3374 【模板】树状数组 1
#pragma GCC optimize(3)
#include<iostream>
#include<cstdio>
using namespace std;
int n,m;
int x,y;
int a[500010];
int opt;
int lowbit(int x){
return x&(-x);
}
void add(int i,int x){
while(i<=n){
a[i]+=x;
i+=lowbit(i);
}
}
int query(int i){
int ans=0;
while(i){
ans+=a[i];
i-=lowbit(i);
}
return ans;
}
int main(){
register int i;
scanf("%d%d",&n,&m);
for(i=1;i<=n;++i){
scanf("%d",&x);
add(i,x);
}
while(m--){
scanf("%d %d %d",&opt,&x,&y);
if(opt^1){
printf("%d\n",query(y)-query(x-1));
}
else{
add(x,y);
}
}
}