树状数组轻度扩展

1.最简单的就不过多赘述,看看别人的文章就行

对于数组a,多次操作加询问,修改其某个点,求其任意区间和

树状数组d[x]保存a的序列区间[x-lowbit(x)+1,x];

二进制k=7=2^{2}+2^{1}+2^{0}

即被分为【1,4】【5,6】【7,7】三个小区间

从1~n,每一个数的前缀和都是被如此·安排

单点修改,区间和查询
ll d[N], n,m;//d为树状数组
//低位运算
ll lowbit(ll x) {
	return x & (-x);
}
//求1~x区间的前缀和
ll Sum(ll x) {
	ll res = 0;
		while (x) {
			res += d[x];
			x -= lowbit(x);
		}
		return res;
}
//单点修改
void add(int i, int z) {
	while (i <= n) {
		d[i] += z;
		i += lowbit(i);
	}
}
//区间和查询
ll query(ll l,ll r){
    return Sum(r)-Sum(l-1);
}
int main() {
	cin >> n;
	ll k;
	for (int i = 1; i <= n;i++) {
		cin >> k;
		add(i, k);
	}
	return 0;
}
区间修改,单点查询

此与上面区别在于,此处维护的是数组的差分数组作为树状数组,进而转化成了问题1

1.查询:设原数组为a[i], 设数组d[i]=a[i]−a[i−1](a[0]=0),则 a[i]=\sum_{j=1}^{i}d[j],可以通过求d[i]的前缀和查询

2.修改:与问题1相同

ll n, a[N], d[N],m;
ll lowbit(ll x) {
	return x & (-x);
}
void add(ll x, ll v) {
	while (x <= n) {
		d[x] += v;
		x += lowbit(x);
	}
}
//使[l,r]区间的值加v
void updata(ll l, ll r, ll v) {
	add(l, v);
	add(r + 1, -v);
}
//单点查询,d的x个前缀和即为x点的值
ll Sum(ll x) {
	ll res = 0;
	while (x) {
		res += d[x];
		x -= lowbit(x);
	}
	return res;
}
int main() {
	cin >> n>>m;
	for (int i = 1; i <= n; i++) {
		cin >> a[i];
		add(i, a[i] - a[i - 1]);
	}
	while (m--) {
		int a, b, c, d;
		cin >> a;
		if (a == 1) {
			cin >> b >> c >> d;
			updata(b, c, d);
		}
		else {
			cin >> b;
			cout << Sum(b) << "\n";
		}
	}
	return 0;
}
区间修改 + 区间查询

我们基于问题2的“差分”思路,考虑一下如何在问题2构建的树状数组中求前缀和:

位置p的前缀和 =

\sum_{i=1}^{p}a[i]=\sum_{i=1}^{p}\sum_{j=1}^{i}d[j]

在等式最右侧的式子\sum_{i=1}^{p}\sum_{j=1}^{i}d[j]中,d[1] 被用了p次,d[2]被用了p−1次……那么我们可以写出:

位置p的前缀和

\sum_{i=1}^{p}\sum_{j=1}^{i}d[j]=\sum_{i=1}^{p}d[i]∗(p−i+1)=(p+1)∗\sum_{i=1}^{p}d[i] − \sum_{i=1}^{p}i∗d[i]

那么我们可以维护两个数组的前缀和:

一个数组是 sum1[i]=d[i]

另一个数组是 sum2[i]=i∗d[i]

ll n, sum[N], c[2][N], m;
ll lowbit(ll x) {return x & (-x);}
void add(ll x, ll v,int k) {
	while (x <= n) {
		c[k][x] += v;
		x += lowbit(x);
	}
}
//一次更新范围,需要维护两个数组
void updata(ll l, ll r, ll v) {
    //对sum1的维护
	add(l, v,0);
	add(r + 1, -v,0);
    //对sum2的维护
	add(l, l*v, 1);
	add(r + 1, -(r+1)*v, 1);
}
//求前缀和
ll Sum(ll x,int k) {
	ll res = 0;
	while (x) {
		res += c[k][x];
		x -= lowbit(x);
	}
	return res;
}
int main() {
	cin >> n >> m;
    //构建初始数组前缀和
	for (int i = 1; i <= n; i++) {
		cin >> sum[i];
		sum[i] += sum[i - 1];
	}
    //操作
	while (m--) {
		int  l,r,d;
		char a;
		cin >> a;
		if (a == 'C') {
		//区间修改
            cin >> l >> r >> d;
			updata(l, r, d);
		}
		else {
         //区间查询
			cin >> l >> r;
         //最后的区间修改值是两个树状数组的区间和的差(上面拆分解释)
         //再将其加上原数组的区间和就是修改后的区间和
         //原数组的区间和sum[r]-sum[l-1]
         // sum1的区间和:(r + 1) * Sum(r, 0) - Sum(r, 1)
		 // sum2的区间和:l*Sum(l-1,0)-Sum(l-1,1
            cout << sum[r]-sum[l-1] + (r + 1) * Sum(r, 0) - Sum(r, 1)-(l*Sum(l-1,0)-Sum(l-1,1)) << "\n";
		}
	}
	return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

骰子.w

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

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

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

打赏作者

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

抵扣说明:

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

余额充值