专栏文章

【笔记】树状数组

算法·理论参与者 1已保存评论 0

文章操作

快速查看文章及其快照的属性,并进行相关操作。

当前评论
0 条
当前快照
1 份
快照标识符
@miow4qgv
此快照首次捕获于
2025/12/03 02:07
3 个月前
此快照最后确认于
2025/12/03 02:07
3 个月前
查看原文
树状数组可以高效率地查询和维护数列的前缀和(或区间和),支持单点修改和区间查询。
根据仍以正整数关于 22 的不重复次幂的唯一分解性质,可以尝试按照二进制拆分,具体如下:
若一个正整数 xx 可以被“二进制分解”成 x=2i1+2i2++2imx=2^{i1}+2^{i2}+\cdots+2^{im},其中 i1>i2>>imi1>i2>\cdots>im,则可以将 [1,x][1,x] 分解为不超过 logx\lceil \log_x\rceil 个小区间:
  1. 长度为 2i12^{i1} 的小区间 [1,2i1][1,2^{i1}]
  2. 长度为 2i22^{i2} 的小区间 [2i1+1,2i1+2i2][2^{i1}+1,2^{i1}+2^{i2}]
  3. \cdots
  4. 长度为 2im2^{im} 的小区间 [2i1+2i2++2im1+1,2i1+2i2++2im][2^{i1}+2^{i2}+\cdots+2^{im-1}+1,2^{i1}+2^{i2}+\cdots+2^{im}]
其中,若区间结尾为 RR,则区间长度为 RR 的二进制分解下最小的 22 的次幂,即 lowbit(R)
由二进制得 lowbit(x)=x&(~x+1)=x&(-x)。因为取反后,末尾的所有 00 变为 11,最低位的 11 变为了最低位的 00,此时再对其加 11,会一直向前进位,直至最低位的 00,此时再进行按位与,自然只有最低位的 11 被保留。当然,因为补码情况下 -x=~x+1,所以还可以进一步简化。
CPP
int lowbit(int x) {return x&(-x);}
int main()
{
  cin >> x;
  for (int i = x; i; i -= lowbit(i))
    //分成的每一个区间为 [i-lowbit(i)+1, i]
}
树状数组基于此,可以维护序列前缀和。对于给定的序列 aa,我们令数组 cc 的第 xxcxc_x 保存序列 aa 的区间 [xlowbit(x)+1,x][x-lowbit(x)+1,x] 中的所有数之和,即 cx=i=xlowbit(x)+1xaic_x=\sum^x_{i=x-lowbit(x)+1}a_i
由此,我们写出树状数组的查询前缀和代码。
CPP
ll query(int x)
{//查询[1,x]区间和
  ll ans=0;
  for(;x>0;x-=x&(-x)) ans += a[x]; //拆区间,计算和
  return ans;
}
事实上,树状数组中,结点 xx 的父亲为 x+lowbit(x)x+lowbit(x)。因此在单点修改中,我们只需要沿着单点(仅管理 axa_x 的叶节点)向根节点的路径修改即可,直到跳到数组长度之外为止。
由此写出树状数组的单点修改代码。
CPP
void add(int x, int y)
{//将 a[x] 加上 y
  for (;x<=n;x+=x&(-x)) a[x]+=y;//不断修改更新并跳向父亲,直至超出数组长度n
}

评论

0 条评论,欢迎与作者交流。

正在加载评论...