HDU 2475 Box 树型转线型 + 舒展树

HDU 2475 Box 树型转线型 + 伸展树

树型转线型,第一次听说这个概念. . . , 但是以前已经接触过了,如LCA的预处理部分和树链剖分等,但是没想到还能这么用,三者虽说有不同但是大体思想还是很相近的,学习了。

推荐博客http://blog.****.net/lyhypacm/article/details/6734748

转成线型之后,就变成了伸展树的模板题。

另外要注意,伸展树的特点是平均时间复杂度接近log(n),所以一定要记得每次操作之后都要伸展。再次学习了。

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <queue>
#include <cmath>
#include <stack>
#include <map>

#pragma comment(linker, "/STACK:1024000000");
#define EPS (1e-8)
#define LL long long
#define ULL unsigned long long
#define _LL __int64
#define _INF 0x3f3f3f3f
#define Mod 9999991
#define lowbit(x) (x&(-x))

using namespace std;

const int MAXN = 50010;

struct N
{
    //info
    int son[2],pre,ls,rs,s;

    //data
    int id;
} st[MAXN*2];

int Top;

void Updata(int root)
{
    st[root].ls = (st[root].son[0] == -1 ? 0 : st[st[root].son[0]].s);
    st[root].rs = (st[root].son[1] == -1 ? 0 : st[st[root].son[1]].s);
    st[root].s = st[root].ls + st[root].rs + 1;
}

void Push_Down(int root)
{
    ;
}

void Rotate(int root,int dir)
{
    st[st[root].pre].son[dir] = st[root].son[1^dir];
    st[root].son[1^dir] = st[root].pre;

    if(st[st[st[root].pre].pre].son[0] == st[root].pre)
        st[st[st[root].pre].pre].son[0] = root;
    else
        st[st[st[root].pre].pre].son[1] = root;
    int temp = st[root].pre;
    st[root].pre = st[st[root].pre].pre;
    st[temp].pre = root;

    if(st[temp].son[dir] != -1)
        st[st[temp].son[dir]].pre = temp;
    Updata(temp);
    Updata(root);
}

int Splay(int root,int goal)
{
    while(st[root].pre != goal)
    {
        Rotate(root,(st[st[root].pre].son[0] == root ? 0 : 1));
    }

    return root;
}

int Search_Site(int root,int site)
{
    Push_Down(root);

    int temp;

    if(st[root].ls + 1 == site)
        temp = root;
    else if(st[root].ls + 1 < site)
        temp = Search_Site(st[root].son[1],site-st[root].ls-1);
    else
        temp = Search_Site(st[root].son[0],site);

    Updata(root);
    return temp;
}

struct E
{
    int v,next;
} edge[MAXN];

int head[MAXN];

int degree[MAXN];

int Top_E;

void Link(int u,int v)
{
    edge[Top_E].v = v;
    edge[Top_E].next = head[u];
    head[u] = Top_E++;
}

int vis[MAXN*2];

int Top_S;

int L[MAXN],R[MAXN];

void dfs(int root)
{
    vis[Top_S++] = root;

    for(int p = head[root]; p != -1; p = edge[p].next)
    {
        dfs(edge[p].v);
    }

    vis[Top_S++] = -root;
}

void NewNode(int root,int id,int pre)
{
    st[root].son[0] = -1,st[root].son[1] = -1;
    st[root].pre = pre;

    st[root].id = id;
}

void Init(int &root,int l,int r,int pre)
{
    if(l > r)
        return ;

    int mid = (l+r)>>1;

    root = Top++;

    NewNode(root,vis[mid],pre);

    if(vis[mid] > 0)
        L[vis[mid]] = root;
    else
        R[-vis[mid]] = root;
    Init(st[root].son[0],l,mid-1,root);
    Init(st[root].son[1],mid+1,r,root);

    Updata(root);
}

int Query(int v)
{
    Splay(L[v],0);
    return st[Search_Site(L[v],1)].id;
}

void Move(int u,int v)
{
    if(u == v)
        return ;
    int site = R[v];

    Splay(R[u],0);
    Splay(L[u],0);

    while(site)
    {
        if(st[site].pre == R[u] && st[R[u]].son[0] == site)
            return ;
        site = st[site].pre;
    }

    if(st[L[u]].son[0] != -1)
    {
        int lt = st[L[u]].son[0];
        int rt = st[R[u]].son[1];

        st[L[u]].son[0] = -1;
        st[R[u]].son[1] = -1;

        Updata(R[u]);
        Updata(L[u]);

        st[lt].pre = 0;

        int root = Splay(Search_Site(lt,st[lt].s),0);
        st[root].son[1] = rt;
        st[rt].pre = root;
        Updata(root);
    }

    if(v)
    {
        Splay(L[v],0);
        Splay(Search_Site(L[v],st[L[v]].ls+2),L[v]);
        st[st[L[v]].son[1]].son[0] = L[u];
        st[L[u]].pre = st[L[v]].son[1];
        Updata(st[L[v]].son[1]);
        Updata(L[v]);
    }
}

int main()
{
    int n,m;

    int i,j,u,v;

    int root;

    bool blank = false;

    while(scanf("%d",&n) != EOF)
    {
        if(blank)
            printf("\n");
        else
            blank = true;
        Top_E = 0;
        memset(head,-1,sizeof(int)*(n+2));
        memset(degree,0,sizeof(int)*(n+2));

        for(i = 1; i <= n; ++i)
        {
            scanf("%d",&u);
            if(u)
            {
                Link(u,i);
                degree[i]++;
            }
        }

        st[0].son[0] = -1,st[1].son[1] = -1;

        Top = 1;

        for(i = 1; i <= n; ++i)
        {
            if(degree[i] == 0)
            {

                Top_S = 1;
                dfs(i);

                root = -1;
                Init(root,1,Top_S-1,0);
            }
        }

        scanf("%d",&m);
        L[0] = R[0] = 0;
        char s[10];
        while(m--)
        {
            scanf("%s",s);
            if(s[0] == 'Q')
            {
                scanf("%d",&u);
                printf("%d\n",Query(u));

            }
            else
            {
                scanf("%d %d",&u,&v);
                Move(u,v);
            }
        }
    }
    return 0;
}