poj3468 線段樹 or splay

poj3468  裸線段樹。由於在熟悉splay 因此就用splay交了一發。。。開始用的scanf()!==2 竟然TLE了。。。而後我就當單組測試數據作的 而後就過了  囧TZ  node

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <cstring>
#include <cmath>
using namespace std;
#define rt ch[root][1]
#define lrt ch[rt][0]
#define ls ch[x][0]
#define rs ch[x][1]
#define inf 0x3f3f3f3f
const int maxn=100100;

int root,top;//root爲新建節點時的根節點
int ch[maxn][2],f[maxn];
long long sz[maxn],num[maxn],val[maxn],sum[maxn],add[maxn];
int n,m;

//ch[x][0],ch[x][1]:x的左右兒子,f[x]:x的父親節點,val[x]:x的值,sz[x]以x爲根的子樹的節點個數
inline void pushup(int x)
{
    sz[x]=sz[ls]+sz[rs]+1;
    sum[x]=sum[ls]+sum[rs]+val[x];
    //維護其餘信息
}
inline void pushdown(int x)
{
    if(add[x])
    {
        val[x]+=add[x];
        add[ls]+=add[x];
        add[rs]+=add[x];
        sum[ls]+=(long long)(sz[ls]*add[x]);
        sum[rs]+=(long long)(sz[rs]*add[x]);
        add[x]=0;
    }
    //維護其餘信息
}

inline void link(int y,int x,int d)
//把節點y的(左或右)兒子設爲x
{
    f[x]=y;
    ch[y][d]=x;
}

inline int is(int x)
//判斷x爲左兒子0仍是右兒子1
{
    return ch[f[x]][1]==x;
}

inline void rotate(int x,int d)
{
    int y=f[x];
    int z=f[y];
    pushdown(y);
    pushdown(x);
    link(y,ch[x][d],!d);
    if(z) link(z,x,is(y));
    f[x]=z;
    link(x,y,d);
    pushup(y);
}
inline void zag(int x)
{
    rotate(x,0);
}
inline void zig(int x)
{
    rotate(x,1);
}
inline void splay(int x,int goal=0)//第二個元素沒有傳參就默認爲0,splay到0即將該節點旋轉到根
//將根爲r的子樹調整到x的父節點爲goal的位置
{
    pushdown(x);

    while(f[x]!=goal)
    {
        int y=f[x];
        int z=f[y];
        if(z==goal)
        {
            rotate(x,!is(x));
            break;
        }
        if(ch[z][0]==y)
        {
            if(ch[y][0]==x) zig(y),zig(x);
            else zag(x),zig(x);
        }
        else
        {
            if(ch[y][1]==x) zag(y),zag(x);
            else zig(x),zag(x);
        }
    }
    if(goal==0) root=x;
    pushup(x);
}

//splay操做通常不作改動
inline void newnode(int &x,int father,long long v)
{
    x=++top;
    f[x]=father;
    val[x]=v;
    add[x]=0;
    ls=rs=0;
    sz[x]=0;

    //.....按題意創建信息
}
inline bool insert(int v)
{
    int x=root;
    while(ch[x][val[x]<v])
    {
        if(val[x]==v)
        {
            splay(x);
            return false;
        }
        x=ch[x][val[x]<v];
    }
    newnode(ch[x][val[x]<v],x,v);
    splay(ch[x][val[x]<v]);
    return true;
}

inline void build(int &x,int y,int l,int r)
{
    if(l>r) return ;
    int mid=(l+r)>>1;
    newnode(x,y,num[mid]);
    build(ls,x,l,mid-1);
    build(rs,x,mid+1,r);
    pushup(x);

}

inline void init(int n)
{
    root=top=0;
    ch[0][0]=ch[0][1]=f[0]=val[0]=add[0]=sz[0]=sum[0]=0;
    newnode(root,0,0);
    newnode(rt,root,0);
    sz[root]=2;
    for(int i=0; i<n; i++) scanf("%lld",&num[i]);
    build(lrt,rt,0,n-1);
    pushup(rt);
    pushup(root);
    //for(int x=0;x<=top;x++){
    //    printf("x=%d f[x]=%d ls=%d rs=%d val=%d sum=%d add=%d sz=%d\n",x,f[x],ls,rs,val[x],sum[x],add[x],sz[x]);
    //}
}

inline int pre(int x)//得到前驅,調用時pre(root)便可
{
    x=ls;
    if(x==0) return inf;
    while(rs) x=rs;
    return val[x];
}


inline int next(int x)//調用時next(root)便可,後繼
{
    x=rs;
    if(x==0) return inf;
    while(ls) x=ls;
    return val[x];
}

inline int get_k(int x,int k)//返回的是第k小的點的標號(從小到大),值要用val來獲取
{
    pushdown(x);
    int num=sz[ls]+1;
    if(num==k) return x;
    if(k<num) return get_k(ls,k);
    return get_k(rs,k-num);
}

inline void update()
{
    int l,r;
    long long c;
    scanf("%d %d %lld",&l,&r,&c);
    int x=get_k(root,l);
    splay(x);
    int y=get_k(root,r+2);
    splay(y,root);
    add[lrt]+=c;
    sum[lrt]+=sz[lrt]*c;
}
//查詢一段區間的值
inline void query()
{
    int l,r;
    scanf("%d%d",&l,&r);
    int x,y;
    x=get_k(root,l);
    y=get_k(root,r+2);
    splay(x);
    splay(y,root);
    printf("%lld\n",sum[lrt]);
}


int main(){
    scanf("%d%d",&n,&m);
    init(n);
    for(int i=1;i<=m;i++){
        int l,r,d;
        char c;
        scanf("%c%c",&c,&c);
        if(c=='Q'){
            //cout<<c<<endl;
            query();
        }
        if(c=='C'){
            update();

        }
    }
}
相關文章
相關標籤/搜索