1. 程式人生 > >【codevs1082】線段樹練習 3

【codevs1082】線段樹練習 3

adl for while small get color ace str 所有

題目描述 Description

給你N個數,有兩種操作:

1:給區間[a,b]的所有數增加X

2:詢問區間[a,b]的數的和。

輸入描述 Input Description

第一行一個正整數n,接下來n行n個整數,

再接下來一個正整數Q,每行表示操作的個數,

如果第一個數是1,後接3個正整數,

表示在區間[a,b]內每個數增加X,如果是2,

表示操作2詢問區間[a,b]的和是多少。

pascal選手請不要使用readln讀入

輸出描述 Output Description

對於每個詢問輸出一行一個答案

樣例輸入 Sample Input

3

1

2

3

2

1 2 3 2

2 2 3

樣例輸出 Sample Output

9

數據範圍及提示 Data Size & Hint

數據範圍

1<=n<=200000

1<=q<=200000

分析

對於線段樹的區間修改時,我們不可能一個一個地把區間內的每一個點都修改了。那樣很耗時,而且有可能有很多點我們用不到。

當要把一個區間的值增加一個數時,我們可以先更新這個區間的值,然後在給這個區間打上一個延遲標記表示下面的區間還未更新,因為下面的區間我們可能用不到。當要用到下面的區間時,就可以將延遲標記向下面傳遞,更新下面區間的值,用得到就更新,用不到就不更新,這就是延遲標記。

總結一下,用延遲標記對區間修改時,被打上標記的區間的值是被更新過的,而下面的區間是未被更新的,因為有可能用不到所以不必更新,當用到下面的區間時就傳遞延遲標記,更新左右區間的值。

代碼

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=200000+5;
typedef long long ll;
inline int read()
{
    int x=0,f=1; char ch=getchar();
    while(ch<0||ch>9){if(ch==-)f=-1; ch=getchar();}
    while(ch>=
0&&ch<=9){x=x*10+ch-0; ch=getchar();} return x*f; } int n,m,len; int a[maxn]; struct node { int l,r,lc,rc,add; ll c; }tr[maxn<<1]; inline void bt(int x,int y) { len++; int now=len; tr[now].l=x; tr[now].r=y; if(x==y) tr[now].c=a[x]; else { int mid=(x+y)>>1; tr[now].lc=len+1; bt(x,mid); tr[now].rc=len+1; bt(mid+1,y); tr[now].c=tr[tr[now].lc].c+tr[tr[now].rc].c; } } inline void pushdown(int now) { int lc=tr[now].lc,rc=tr[now].rc; tr[lc].c+=tr[now].add*(tr[lc].r-tr[lc].l+1); tr[rc].c+=tr[now].add*(tr[rc].r-tr[rc].l+1); tr[lc].add+=tr[now].add; tr[rc].add+=tr[now].add; tr[now].add=0; } inline void update(int now,int x,int y,int k) { if(tr[now].l==x&&tr[now].r==y) { tr[now].c+=k*(tr[now].r-tr[now].l+1); tr[now].add+=k; }else { int lc=tr[now].lc,rc=tr[now].rc; pushdown(now); int mid=(tr[now].l+tr[now].r)>>1; if(y<=mid) update(lc,x,y,k); else if(x>=mid+1) update(rc,x,y,k); else {update(lc,x,mid,k); update(rc,mid+1,y,k);} tr[now].c=tr[lc].c+tr[rc].c; } } inline ll query(int now,int x,int y) { if(tr[now].l==x&&tr[now].r==y) return tr[now].c; else { int lc=tr[now].lc,rc=tr[now].rc; pushdown(now); int mid=(tr[now].l+tr[now].r)>>1; if(y<=mid) return query(lc,x,y); else if(x>=mid+1) return query(rc,x,y); else return query(lc,x,mid)+query(rc,mid+1,y); } } int main() { n=read(); for(int i=1;i<=n;i++) a[i]=read(); bt(1,n); m=read(); for(int i=1;i<=m;i++) { int p,x,y,k; p=read(); if(p==1) { x=read();y=read();k=read(); update(1,x,y,k); }else if(p==2) { x=read();y=read(); printf("%lld\n",query(1,x,y)); } } return 0; }

【codevs1082】線段樹練習 3