zoukankan      html  css  js  c++  java
  • HDU 2475 Box 树型转线型 + 伸展树

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

    推荐博客http://blog.csdn.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(" "); 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 ",Query(u)); } else { scanf("%d %d",&u,&v); Move(u,v); } } } return 0; }


  • 相关阅读:
    透视表提取不反复记录(1)-出现值
    ORA-38760: This database instance failed to turn on flashback database
    Android蓝牙串口程序开发
    指尖上的电商---(5)schema.xml配置具体解释
    iOS-UIImage imageWithContentsOfFile 和 imageName 对照
    JSON-RPC轻量级远程调用协议介绍及使用
    POJ 2296 Map Labeler(2-sat)
    接口測试-HAR
    [Leetcode]Combination Sum II
    MarkDown、Vim双剑合璧
  • 原文地址:https://www.cnblogs.com/clnchanpin/p/7057921.html
Copyright © 2011-2022 走看看