树状数组
**前言:**树状数组现在多用于解决单点修改,求前缀和工作,时间复杂度均为 O ( l o g n ) O(logn) O(logn),空间复杂度为 O ( n ) O(n) O(n)。相比较于线段树,树状数组的作用就显得比较局限,但是它的优点就在于它的常数比较小。线段树在未优化的情况下空间复杂度为 O ( 2 n ) O(2n) O(2n),有时候为了防治溢出往往要开长度为 4 n 4n 4n的数组。
树状数组基本思想
树状数组主要依赖于任何数都可以转化成 2 i 2^i 2i相加的形式,例如 7 = 2 2 + 2 1 + 2 0 7=2^2+2^1+2^0 7=22+21+20,并且如果按照指数从大到小进行区间划分,那么 [ 1 − 7 ] [1-7] [1−7]这个区间可以划分成 [ 1 − 4 ] , [ 5 − 6 ] , [ 7 , 7 ] [1-4],[5-6],[7,7] [1−4],[5−6],[7,7]这三个区间。
即有整数 x = 2 i 1 + 2 i 2 + . . . + 2 i j x=2^{i_1}+2^{i_2}+...+2^{i_j} x=2i1+2i2+...+2ij,可以将区间划分成: [ 1 − 2 i 1 ] , [ 2 i 1 + 1 , 2 i 2 ] . . . [ 2 i 1 + . . . 2 i j − 1 + 1 , 2 i j ] [1-2^{i_1}],[2^{i_1}+1,2^{i_2}]...[2^{i_1}+...2^{i_{j-1}}+1,2^{i_j}] [1−2i1],[2i1+1,2i2]...[2i1+...2ij−1+1,2ij],共 l o g n logn logn个区间,并且我们发现每个区间的区间长度恰好为 l o w b i t ( 2 i ) lowbit(2^i) lowbit(2i)
l o w b i t ( x ) lowbit(x) lowbit(x)指的是 x x x在二进制中,最低位的 1 1 1与它后面的 0 0 0进行的组合。
树状数组的实现
树状数组总共有两个操作,首先是修改操作。定义一个数组 c [ x ] c[x] c[x],表示从 x − l o w b i t ( x ) + 1 , x x-lowbit(x)+1,x x−lowbit(x)+1,x的和。当在第 w w w位上加上 n u m num num,那么就要对它后面的数进行更改,它后面的数就是 l o w b i t ( i ) lowbit(i) lowbit(i)的长度
void add(int w,int num){
for(int i=w;i<=n;i+=i&(-i)){
c[i]+=num;
}
}
查询操作
int query(int x){
int ans=0;
for(int i=x;i>=1;i-=i&(-i)){
ans+=c[i];
}
return ans;
}