【模板】线段树

今天考试的时候很开心的打了一个线段树,考完一看,SMG

#include<iostream> 
#include<cstring>
#include<cstdio>
#include<algorithm>
#define maxn 100001
#define LL long long 
using namespace std;
struct node
{LL l,r,sum;}tree[maxn*3];
LL a[maxn],f[maxn],lazy[maxn*3];
inline LL read()
{
    char ch=getchar();
    LL f=1,x=0;
    while(!(ch>='0'&&ch<='9')) { if(ch=='-') f=-1;ch=getchar();}
    while((ch>='0'&&ch<='9')) {x=x*10+(ch-'0');ch=getchar();}
    return f*x;
}
void build(LL x,LL l,LL r)
{
    tree[x].l=l;tree[x].r=r;
    if(l==r){tree[x].sum=read();return;}
    build(x*2,l,(l+r)>>1);
    build(x*2+1,1+((l+r)>>1),r);
    tree[x].sum=tree[x*2+1].sum+tree[x*2].sum;
}
void pushdown(LL x)
{
    LL mid=(tree[x].l+tree[x].r)/2;
    tree[x*2].sum+=(mid-tree[x].l+1)*lazy[x];
    tree[x*2+1].sum+=(tree[x].r-mid)*lazy[x];
    lazy[x*2]+=lazy[x];
    lazy[x*2+1]+=lazy[x];
    lazy[x]=0;
}
void update(LL x,LL l,LL r,LL k)
{
    if(tree[x].l>=l&&tree[x].r<=r)
    {
        tree[x].sum+=(tree[x].r-tree[x].l+1)*k;
        lazy[x]+=k;
        return;
    }
    if(tree[x].l>r||tree[x].r<l) return;
    LL mid=(tree[x].r+tree[x].l)>>1;
    if(lazy[x]) pushdown(x);
    update(x*2,l,r,k);update(x*2+1,l,r,k);
    tree[x].sum=tree[x*2].sum+tree[x*2+1].sum;
}
LL query(LL x,LL l,LL r)
{
    if(tree[x].l>=l&&tree[x].r<=r) return tree[x].sum;
    if(tree[x].l>r||tree[x].r<l) return 0;
    if(lazy[x]) pushdown(x);
    return query(x*2,l,r)+query(2*x+1,l,r);
}
int main()
{
   LL n,m,x,y,z;
   n=read();m=read();
   build(1,1,n);
   for(LL i=1;i<=m;i++)
   {
        z=read();x=read();y=read();
        if(z==1) {z=read();update(1,x,y,z);}
        else cout<<query(1,x,y)<<endl;
   }
   return 0; 
}
View Code

 

posted @ 2016-11-16 15:03  Native_carrot  阅读(148)  评论(0编辑  收藏  举报