与图论的邂逅04:LCT

  本着对数据结构这一块东西的一股兴趣,最近在集训的百忙之中抽空出来学LCT,终于学懂了这个高级玩意儿。


前置知识:Splay和树链剖分

  Splay挺复杂的......这里就先不写,不然篇幅太大。树链剖分倒是可以大致地讲一下。

树链剖分

  什么是树链剖分呢?就是把树给解剖成一条条的链子啦~那就先从最常用的重链剖分讲起。对于当前节点u和它的儿子构成的点集V,若size[v]=max{size[w] | w∈V},也就是以u的儿子为根的所有子树中size最大的那个儿子是v,那么称v是u的重儿子(定义size[x]:以x为根的子树所含有的点数)。如果这时把u向v连一条边,并且其它所有点也像这么做(连向它们的重儿子,叶子节点除外),由于重儿子唯一,父亲唯一,所以就会形成一条一条的链。这个链就叫重链。重链剖分是如此,类比一下就能够理解长链剖分:每个点连向子树深度最大的那个儿子。然而这两种剖法都不是LCT所使用的。LCT使用的是:实链剖分。

  实链剖分就是说,把某个节点向它的某个儿子连实边,向其它儿子连虚边,这样也能连出一条条链子。实链剖分有什么好处呢?重链剖分和长链剖分中的size和dep都是固定的,也就是说剖完一棵树之后就无法再改变。而注意刚才实链剖分定义中的一句话:"把某个节点向它的某个儿子连实边",也就是说这个"某个儿子"是可以改变的。那么实链剖分就应运而生并用于解决更灵活的问题。

  如果你觉得还不清楚,你可以看一下苯蒟蒻对重链剖分一点点浅显的理解:https://www.cnblogs.com/akura/p/10692600.html,然后类比地这些链剖分应该就可以理解了。


LCT

  LCT使用Splay作为辅助树来维护实链,其Splay平衡的关键字为深度。也就是说,右边的点深度大于根,左边的点深度小于根。下面是LCT的相关操作:

access(x)

  指打通x到根节点之间的路径,即建立一条从根节点连向x的实链。只需要从x一个链子一个链子地往上跳,并且把跳到的链子改一下即可。原本x到根节点路径上的点可能属于其它的实链,也就是说它的儿子不在x到根节点的路径上——这时把这对父子关系的边断掉即可:

 

  图中粗的是Splay维护的实链,细的是轻边。access之后成了这个样子:

 

  也就是说,我往上连链子的时候碰到一个点,我不管它之前连的哪条链子,都给他断掉,连上我要的链子。

inline void access(int x){ for(int y=0;x;ch[x][1]=y,update(x),x=fa[y=x]) splay(x); }

  每次把x旋到当前Splay的根节点处,往上连边。由于改变了Splay,所以要update一下。

makeroot(x)

  指让x成为当前的根。很简单,只需打通x到根节点的路径,然后用splay把x旋到链子顶端即可。最后不要忘了reverse(因为dep发生了改变)。

  这是access之后x和根节点之间的实链;

 

 

   这是splay了并且reverse过的这条链子。

inline void makeroot(int x){ access(x),splay(x),rev(x); }
//rev为reverse

split(x,y)

  指打通x到y的路径。先用makeroot(x)让x成为树根,然后用access(y)打通y到根节点(也就是x)的路径即可。最后为了保证复杂度,把y节点splay一下。

 

  然后makeroot(x)。

 

  再access。

  最后splay一下,就不画了。(splay的部分都不画了吧......太麻烦)

inline void split(int x,int y){ makeroot(x),access(y),splay(y); }

findroot(x)

  寻找x所在原树的树根。先access(x)打通根到x的路径,然后splay把x旋到Splay的根。此时树的形态发生改变,所以要下传标记。由于根的深度是最小的,所以此时从x开始往Splay的左儿子一直走,最后走到的点就是根。为了保证复杂度,最后再splay一下根。

inline int findroot(int x){ access(x),splay(x),down(x); while(ch[x][0]) x=ch[x][0],down(x); return splay(x),x; }

link(x,y)

  从x往y连一条边。首先makeroot(x)让x成为根,此时link(x,y)是合法的当且仅当findroot(y)!=x即y所在原树的根不是x。这时只需让fa[x]=y即可。不要忘了update(y)更新一下儿子的信息。

inline void link(int x,int y){ makeroot(x); if(findroot(y)!=x) fa[x]=y,update(y); }

cut(x,y)

  切断x与y之间的边。首先还是makeroot(x)让x成为根,此时cut(x,y)是合法的当且仅当findroot(y)=x并且fa[y]=x并且ch[y][0]=0。最后一个条件是什么意思呢?由于Splay中左儿子的深度小于根,所以若ch[y][0]不为0,则说明还有比y深度更小的点介于x与y之间,此时不能删边。判断合法之后,修改两个节点的信息,令ch[x][1]=0,fa[y]=0即可。还是不要忘了更新一下x的儿子的信息。

inline void cut(int x,int y){ makeroot(x); if(findroot(y)==x&&fa[y]==x&&!ch[y][0]) ch[x][1]=fa[y]=0,update(x); }

  以上就是LCT的最基础的操作(苯蒟蒻还没有学其它的高级操作)。难理解的部分都画了图,应该是能看懂的。

  最后对着模板题放个代码:

给定n个点以及每个点的权值,要你处理接下来的m个操作。操作有4种。操作从0到3编号。点从1到n编号。

0:后接两个整数(x,y),代表询问从x到y的路径上的点的权值的xor和。保证x到y是联通的。

1:后接两个整数(x,y),代表连接x到y,若x到y已经联通则无需连接。

2:后接两个整数(x,y),代表删除边(x,y),不保证边(x,y)存在。

3:后接两个整数(x,y),代表将点x上的权值变成y。

#include<iostream>
#include<cstring>
#include<cstdio>
#define maxn 300001
using namespace std;
typedef long long ll;

inline int read(){
    register int x(0),f(1); register char c(getchar());
    while(c<'0'||'9'<c){ if(c=='-') f=-1; c=getchar(); }
    while('0'<=c&&c<='9') x=(x<<1)+(x<<3)+(c^48),c=getchar();
    return x*f;
}
inline void swap(ll &a,ll &b){ register int tmp; tmp=a,a=b,b=tmp; }

int  ch[maxn][2],fa[maxn],lz[maxn];
ll sum[maxn],val[maxn];

inline int isroot(int x){ return ch[fa[x]][0]==x||ch[fa[x]][1]==x; }
inline int getson(int x){ return ch[fa[x]][1]==x; }
inline void update(int x){ sum[x]=val[x]^sum[ch[x][0]]^sum[ch[x][1]]; }
inline void rev(int x){ lz[x]^=1; swap(ch[x][0],ch[x][1]); }
inline void down(int x){ if(lz[x]){ if(ch[x][0]) rev(ch[x][0]); if(ch[x][1]) rev(ch[x][1]); } lz[x]=0; }
inline void rotate(int x){
    int y=fa[x],z=fa[y],k=getson(x),w=ch[x][k^1];
    if(isroot(y)) ch[z][getson(y)]=x;fa[x]=z;
    ch[x][k^1]=y,fa[y]=x;
    if(w) fa[w]=y; ch[y][k]=w;
    update(y);
}
inline void splay(int x){
    int y,z,stack[maxn]; y=x,z=0,stack[++z]=y;
    while(isroot(y)) y=fa[y],stack[++z]=y; while(z) down(stack[z--]);
    for(y=fa[x]; isroot(x); rotate(x),y=fa[x]) if(isroot(y)) rotate(getson(y)^getson(x)?x:y); update(x);
}

inline void access(int x){ for(int y=0;x;ch[x][1]=y,update(x),x=fa[y=x]) splay(x); }
inline void makeroot(int x){ access(x),splay(x),rev(x); }
inline void split(int x,int y){ makeroot(x),access(y),splay(y); }
inline int findroot(int x){ access(x),splay(x),down(x); while(ch[x][0]) x=ch[x][0],down(x); return splay(x),x; }
inline void link(int x,int y){ makeroot(x); if(findroot(y)!=x) fa[x]=y,update(y); }
inline void cut(int x,int y){ makeroot(x); if(findroot(y)==x&&fa[y]==x&&!ch[y][0]) ch[x][1]=fa[y]=0,update(x); }

int main(){
    int n=read(),m=read();
    for(register int i=1;i<=n;i++) sum[i]=val[i]=read();
    for(register int i=1;i<=m;i++){
        int op=read(),x=read(),y=read();
        if(op==0) split(x,y),printf("%lld\n",sum[y]);
        if(op==1) link(x,y);
        if(op==2) cut(x,y);
        if(op==3) splay(x),val[x]=y;
    }
    return 0;
}

 

posted @ 2019-04-29 09:12  修电缆的建筑工  阅读(220)  评论(0编辑  收藏  举报