题目描述
如题,已知一个数列,你需要进行下面两种操作:
将某一个数加上 xxx
求出某区间每一个数的和
输入格式
第一行包含两个正整数 n,mn,mn,m,分别表示该数列数字的个数和操作的总个数。
第二行包含 nnn 个用空格分隔的整数,其中第 iii 个数字表示数列第 iii 项的初始值。
接下来 mmm 行每行包含 333 个整数,表示一个操作,具体如下:
1 x k 含义:将第 xxx 个数加上 kkk
2 x y 含义:输出区间 [x,y][x,y][x,y] 内每个数的和
输出格式
输出包含若干行整数,即为所有操作 222 的结果。
输入输出样例
输入 #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≤81 \le n \le 81≤n≤8,1≤m≤101\le m \le 101≤m≤10;
对于 70%70%70% 的数据,1≤n,m≤1041\le n,m \le 10^41≤n,m≤104;
对于 100%100%100% 的数据,1≤n,m≤5×1051\le n,m \le 5\times 10^51≤n,m≤5×105。
#include <iostream>
#include <algorithm>
#include <cstdio>
using namespace std;
typedef long long ll;
const int maxn = 5e5 + 10;
int a[maxn];
ll t[maxn];
int n, m;
int lowbit(int x) {
return x & -x;
}
void update(int id, int val) {
for(; id <= n; id += lowbit(id)) {
t[id] += val;
}
}
ll query(int i) {
ll sum = 0;
for(; i >= 1; i -= lowbit(i)) {
sum += t[i];
}
return sum;
}
int main() {
cin >> n >> m;
for(int i = 1 ;i <= n;i ++)
cin >> a[i], update(i, a[i]);
int id, x, y;
for(int i = 1; i <= m; i++) {
cin >> id >> x >> y;
if(id == 1)
update(x, y);
else
// cout << query(y) << " " << query(x - 1) << endl;
cout << query(y) - query(x - 1) << endl;
}
return 0;
}