树状数组区间修改

Posted 123456

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了树状数组区间修改相关的知识,希望对你有一定的参考价值。

codevs 1082

白书上很详尽。维护两个东西,一个是差量,也就是哪一段被更新了多少,还有一个也是差量,只不过是记录一共减了多少。(说不清)

然后根据白书上的式子,自己yy一下。还是不是很懂

#include<cstdio>
#include<cstring>
using namespace std;
#define lowbit(x) x&-x
#define N 200010
typedef long long ll;
int n,q;
ll sum[N],t1[N],t2[N];
void add1(int pos,int delta)
{
    for(int i=pos;i<=n;i+=lowbit(i))
        t1[i]+=delta;
}
void add2(int pos,int delta)
{
    for(int i=pos;i<=n;i+=lowbit(i))
        t2[i]+=delta;
}
ll sum1(int pos)
{
    ll ret=0;
    for(int i=pos;i>0;i-=lowbit(i))
        ret+=t1[i];
    return ret;
}
ll sum2(int pos)
{
    ll ret=0;
    for(int i=pos;i>0;i-=lowbit(i))
        ret+=t2[i];
    return ret;
}
int main()
{
    scanf("%d",&n);
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&sum[i]);
        sum[i]+=sum[i-1];
    }
    scanf("%d",&q);
    while(q--)
    {
        int opt,a,b; scanf("%d%d%d",&opt,&a,&b);
        if(opt==1)
        {
            int x; scanf("%d",&x);
            add1(a,-(a-1)*x);
            add1(b+1,b*x);
            add2(a,x);
            add2(b+1,-x);
        }
        else 
        {
            printf("%lld\n",sum[b]-sum[a-1]+sum1(b)-sum1(a-1)+sum2(b)*b-sum2(a-1)*(a-1));
        }
    }
    return 0;
}

 

以上是关于树状数组区间修改的主要内容,如果未能解决你的问题,请参考以下文章

树状数组区间修改和区间求和

树状数组从入门到弃疗

树状数组

P3374 模板树状数组 1(单点修改区间查询)(树状数组)

区间修改区间查询(树状数组)

P3372 模板线段树 1(区间修改区间查询)(树状数组)