树状数组的实现

int bit[maxn];//别忘了初始化为0
//在bit数组中需要加上x的地方加上x,使得用sum函数查询的时候显示出[i~maxn)区间内所有的元素都加上x
void add(int i,int x){
	while(i<=n){
		bit[i]+=x;
		i += (i & -i);
	}
}
//利用bit数组的特性求[1~i]这些位置上的数的和
int sum(int i){
	int s=0;
	while(i>0){
		s += bit[i];
		i -= (i & -i);
	}
	return s;
}

//二维上的二维数组和此相似
int bit[maxn][maxn];
int lowbit(int i){
	return i & -i;
}
void add(int x,int y,int v){
	for(int i=x;i<=n;i+=lowbit(i))
		for(int j=y;j<=n;j+=lowbit(j))
			bit[i][j]+=v;
}
int sum(int x,int y){
	int s=0;
	for(int i=x;i>0;i-=lowbit(i))
		for(int j=y;j>0;j-=lowbit(j))
			s += bit[i][j];
	return s;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
树状数组是一种用于快速维护数组前缀和的数据结构,其时间复杂度为 $O(logn)$。树状数组实现基于二进制的思想,借助树状数组可以高效地进行单点更新和区间查询等操作。以下是树状数组实现原理: 1. 数组的每个元素代表原数组中某个位置的前缀和,如下所示: $$C_i = A_1 + A_2 + \cdots + A_i$$ 其中 $C_i$ 表示原数组 $A$ 的前 $i$ 项之和。 2. 将 $C$ 数组转换成树状数组 $T$,树状数组的每个节点表示其父节点到该节点的元素和。如下所示: $$T_i = C_{i-lowbit(i)+1} + C_{i-lowbit(i)+2} + \cdots + C_i$$ 其中 $lowbit(i)$ 表示 $i$ 的二进制表示中最低位的 $1$ 所代表的值。 3. 树状数组的单点更新操作只需要将 $i$ 位置的值加上 $k$,然后依次更新其祖先节点即可: ```cpp void update(int i, int k, int n, vector<int>& tree) { while (i <= n) { tree[i] += k; i += lowbit(i); } } ``` 其中 $n$ 表示原数组 $A$ 的大小,$tree$ 表示树状数组。 4. 树状数组的区间查询操作只需要通过两次前缀和的差值计算出区间和即可: ```cpp int query(int i, vector<int>& tree) { int sum = 0; while (i > 0) { sum += tree[i]; i -= lowbit(i); } return sum; } int query(int i, int j, vector<int>& tree) { return query(j, tree) - query(i - 1, tree); } ``` 其中第一个查询函数用于计算 $A_1 + A_2 + \cdots + A_i$,第二个查询函数用于计算 $A_{i+1} + A_{i+2} + \cdots + A_j$,然后两者相减即可得到区间 $[i,j]$ 的和。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

门豪杰

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值