数据结构——树状数组

树状数组

**前言:**树状数组现在多用于解决单点修改,求前缀和工作,时间复杂度均为 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] [17]这个区间可以划分成 [ 1 − 4 ] , [ 5 − 6 ] , [ 7 , 7 ] [1-4],[5-6],[7,7] [14],[56],[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}] [12i1],[2i1+1,2i2]...[2i1+...2ij1+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 xlowbit(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;
}
相关题目

P3374 【模板】树状数组 1

P1908 逆序对

P1972 [SDOI2009] HH的项链

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值