码迷,mamicode.com
首页 > 其他好文 > 详细

【模板】LCT

时间:2019-12-07 14:28:58      阅读:92      评论:0      收藏:0      [点我收藏+]

标签:退出   shu   支持   不能   access   中序遍历   har   pre   实现   

LCT:

动态维护一个森林。支持删边,加边,查询链信息等很多操作。

由若干棵$Splay$组成,每棵$Splay$维护一条链,以深度作为关键字。

也就是说$Splay$的中序遍历相当于从上到下遍历这条链。

$Splay$中的边是实边,将两个$Splay$相连的边是虚边。

实边的父亲有它这个儿子(双向关系),虚边的父亲没有它这个儿子(单向关系)。

 

组成$LCT$的基础操作:

(以下均认为$LCT$中只有一棵树)

$access(x)$:打通根到$x$的路径,使一棵包含且仅包含根到$x$这条链上点的$Splay$出现。

实现方法:

1.$splay(x)$:将$x$转到当前$Splay$的根。(此时$x$是该$Splay$中最深的点,没有右儿子)

2.$c[x][1]=y$:将$x$的右儿子设为刚才操作的$Splay$的根。

3.$x=f[x]$:继续操作$x$在原树中的父亲,若$x$已经为根则退出。

 

$makeroot(x)$:使$x$成为根。

实现方法:

1.$access(x)$。

2.$splay(x)$。

3.翻转整棵$Splay$。

为什么不能只翻转$x$的左右儿子:一个链提末端点当根之后整个链的深度顺序全部翻转。

如:$1-2-3$翻转后为$3-2-1$而不是$3-1-2$。

 

$findroot(x)$:找$x$所在原树的根。

实现方法:

1.$access(x)$。

2.$splay(x)$。

3.在$Splay$上一直往左走,最终走到答案点$y$。

3.$splay(y)$。

 

$split(x,y)$:拉出路径$(x,y)$成为一个$Splay$,并令$x$为原树的根。

实现方法:

1.$makeroot(x)$。

2.$access(y)$。

 

$link(x,y)$:连一条边$(x,y)$。

实现方法:

1.$makeroot(x)$。

2.若$findroot(y)\neq x$则$f[x]=y$。

 

$cut(x,y)$:断开边$(x,y)$。

实现方法:

1.若$findroot(x)=findroot(y)$则$split(x,y)$。

2.若$f[y]=x$且$c[y][0]=0$则$f[y]=c[x][1]=0$。

 

代码:

技术图片
#include<bits/stdc++.h>
#define maxn 100005
#define maxm 500005
#define inf 0x7fffffff
#define ll long long

using namespace std;
int N,M,rc,A[maxn],s[maxn],st[maxn];
int r[maxn],c[maxn][2],f[maxn];

inline int read(){
    int x=0,f=1; char c=getchar();
    for(;!isdigit(c);c=getchar()) if(c==-) f=-1;
    for(;isdigit(c);c=getchar()) x=x*10+c-0;
    return x*f;
}

inline bool nroot(int x){return c[f[x]][0]==x||c[f[x]][1]==x;}
inline void pushr(int x){swap(c[x][0],c[x][1]),r[x]^=1;}
inline void pushup(int x){s[x]=s[c[x][0]]^s[c[x][1]]^A[x];}
inline void pushdown(int x){
    if(r[x]){
        if(c[x][0]) pushr(c[x][0]);
        if(c[x][1]) pushr(c[x][1]);
        r[x]=0;
    }
}
inline void rotate(int x){
    int y=f[x],z=f[y],k=(c[y][1]==x),w=c[x][!k];
    if(nroot(y)) c[z][c[z][1]==y]=x;
    c[x][!k]=y,c[y][k]=w;
    if(w) f[w]=y;
    f[y]=x,f[x]=z;
    pushup(y);
}
inline void splay(int x){
    int y=x,z=0; st[++z]=y;
    while(nroot(y)) st[++z]=y=f[y];
    while(z) pushdown(st[z--]);
    while(nroot(x)){
        y=f[x],z=f[y];
        if(nroot(y)) rotate(((c[y][0]==x)^(c[z][0]==y))?x:y);
        rotate(x);
    }
    pushup(x);
}
inline void access(int x){for(int y=0;x;x=f[y=x]) splay(x),c[x][1]=y,pushup(x);}
inline void makeroot(int x){access(x),splay(x),pushr(x);}
inline void split(int x,int y){makeroot(x),access(y),splay(y);}
inline int findroot(int x){
    access(x),splay(x);
    while(c[x][0]) pushdown(x),x=c[x][0];
    splay(x); return x;
}
inline void link(int x,int y){makeroot(x);if(findroot(y)!=x) f[x]=y;}
inline void cut(int x,int y){makeroot(x);if(findroot(y)==x && f[y]==x && !c[y][0])f[y]=c[x][1]=0,pushup(x);}

int main(){
    N=read(),M=read();
    for(int i=1;i<=N;i++) A[i]=read();
    while(M--){
        int op=read(),x=read(),y=read();
        if(op==0) split(x,y),printf("%d\n",s[y]);
        if(op==1) link(x,y);
        if(op==2) cut(x,y);
        if(op==3) splay(x),A[x]=y;
    } 
    return 0;
}
LCT

 

【模板】LCT

标签:退出   shu   支持   不能   access   中序遍历   har   pre   实现   

原文地址:https://www.cnblogs.com/YSFAC/p/12001161.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!