基本原理
树状数组的原理简单来说就是利用二进制拆分区间
我们可以对一个数进行二进制分解,最多分解成log(x)个数,同样我们可以对[1,n]这个区间进行分解。也是最多log段,每次修改时我们维护受到影响的区间,然后查询时用这log个区间拼凑出一个前缀。这就是树状数组的大概思想。
最基本的作用是动态维护前缀和
在定义树状数组时,我们定义
c
[
i
]
数组
c[i]数组
c[i]数组
c
[
x
]
=
∑
i
=
x
−
l
o
w
b
i
t
(
x
)
+
1
x
a
[
i
]
即
c
[
i
]
保存的时
[
x
−
l
o
w
b
i
t
(
x
)
+
1
,
x
]
中所有数的和
c[x]=\sum_{i=x-lowbit(x)+1}^xa[i] \quad 即c[i]保存的时[x-lowbit(x)+1,x]中所有数的和
c[x]=∑i=x−lowbit(x)+1xa[i]即c[i]保存的时[x−lowbit(x)+1,x]中所有数的和
重点:
c
[
x
]
管辖区间的长度是多少?
c[x]管辖区间的长度是多少?
c[x]管辖区间的长度是多少?
操作
1.查询
x-=lowbit(x)就是下次个要跳到的地方
int ask(int x)
{
int sum=0;
auto lowbit=[](x){return x&-x;};
for(;x;x-=lowbit(x)) sum+=c[x];
return sum;
}
2.修改
树状数组修改时要将,从当前点到根节点这以路径上的所有点都进行修改
void add(int x,int y)
{
for(x;x<=n;x+=lowbit(x)) c[x]+=y;
}
建树过程就看成n次修改过程时间复杂度为 O ( n l o g n ) O(nlogn) O(nlogn)