LCT(Link-Cut Tree,动态树)是一个支持动态修改树的结构的数据结构,其基本操作有 \(\texttt{access}\) , \(\texttt{findroot}\) , \(\texttt{makeroot}\), 其核心操作有 \(\texttt{link}\) 和 \(\texttt{cut}\) .

本文具体讲述 LCT 的操作内容和实际运用,不深入讨论如何得到每个具体操作。

1. LCT 简介

这里用到树链剖分的思想。LCT 的本质是动态维护链剖分。我们将树链剖分,分为实链和虚链,每一条实链用辅助树即平衡树(选择splay)来维护链上信息,每一条虚链连接各个辅助树,每棵辅助树按照深度为关键字维护。每次我们动态进行实链剖分来完成对树上信息和结构的操作。

2. LCT 的实现

接下来介绍 LCT 的各种操作:

2.1. ACCESS

\(\texttt{access(x)}\) 是 LCT 最基本的操作,功能为将根到 \(x\) 的一条链变为实链。这样的用途当然是将这段路径放入一棵辅助树,方便进行操作。

这里不再手玩,直接给出操作过程 (设 \(y\) 为 \(x\) 新实链上的儿子):

  1. 将 \(x\) 旋转到当前辅助树的根
  2. 将 \(x\) 辅助树中的右儿子改为 \(y\) 。
  3. \(y \leftarrow x, x\leftarrow fa[x]\) ,向上递归操作。

具体可以这样理解:将 \(x\) splay 后,他的右儿子比 \(x\) 的深度大,即为一条实链,那我们就断掉这条实链,把 \(y\) 接上去,具体实现即为直接修改右儿子。

for(int y=0;x;y=x,x=fa[x]) splay(x),ch[x][1]=y,pushup(x);

2.2. MAKEROOT

\(\texttt{makeroot(x)}\) 即将 \(x\) 提到原树的根。具体操作:我们 \(\texttt{access(x),splay(x)}\) 后,\(x\) 变为当前辅助树的根,且由于其深度最大,所以没有右子树。然后我们把左右子树翻转,这样就没有比 \(x\) 深度更小的点了, \(x\) 就变成根啦。

access(x),splay(x),rev[x]^=1;

2.3. FINDROOT

\(\texttt{findroot(x)}\) 即为找 \(x\) 所在原树的树根(注意:LCT维护的不一定是一棵树,可能是森林!)。

具体操作: \(\texttt{access(x),splay(x)}\) 后,一路走到最左边即可。很好理解。当然要记得 \(\texttt{pushdown}\) .

特别注意:最后要 splay(x) 保证复杂度!!!

access(x),splay(x); int y=x;
while(ch[y][0]) pushdown(y),y=ch[y][0];
splay(x); return y;

(PS:这可能是LCT基本操作里最长的函数了……)

(PS: 在只有 link 操作的题目可能用并查集更简单)

2.4. LINK

\(\texttt{link(x,y)}\) 即为连接两个节点。直接 \(\texttt{makeroot(x)}\) 后将 \(x\) 的父亲变为 \(y\) 即可。

在 \(\texttt{link}\) 操作不保证合法的情况下,还需要判断 \(y\) 的根是否为 \(x\) 。(代码未涉及)

makeroot(x),fa[x]=y;

2.5. SPLIT

不算是基本操作,但也很常用且简单。 \(\texttt{split(x,y)}\) 即为将 \(x\rightarrow y\) 的一条路径拉出来。我们将 \(x\) 置为根后 \(\texttt{access(y)}\) 即可。再加上 \(\texttt{splay(y)}\) 即可直接操作 \(y\) ,\(x\) 即为 \(y\) 的左儿子。

makeroot(x),access(y),splay(y);

2.6. CUT

\(\texttt{cut(x,y)}\) 即为断掉 \(x\rightarrow y\) 的这条边。 \(\texttt{split(x,y)}\) 之后直接将 \(fa[x]\) 和 \(ch[y][0]\) 置为 \(0\) 即可。

split(x,y),fa[x]=ch[y][0]=0;

在 \(\texttt{cut}\) 操作不保证合法的情况下,在 \(\texttt{makeroot}\) 后需判断:

if(findroot(y)==x&&f[y]==x&&!c[y][0]) do...

3. 辅助树

辅助树之前说过了,用splay来实现。接下来具体讲一讲 LCT 中的 splay 中的变化。

这里先直接贴代码:

#define isnrt(x) (ch[fa[x]][0]==x||ch[fa[x]][1]==x)
void pushdown(int x)
{
if(rev[x])
{
rev[ch[x][0]]^=1,rev[ch[x][1]]^=1;
swap(ch[x][0],ch[x][1]);
rev[x]=0;
}
}
void rotate(int x){
int y=fa[x],z=fa[y];
bool k=ch[y][0]==x; int w=ch[x][k];
if(isnrt(y))ch[z][ch[z][1]==y]=x;ch[x][k]=y;ch[y][!k]=w;
fa[w]=y;fa[y]=x;fa[x]=z;
}
void splay(int x)
{
int y=x,tp=1; st[1]=y;
while(isnrt(y)) st[++tp]=y=fa[y];
while(tp) pushdown(st[tp--]);
while(isnrt(x))
{
int y=fa[x],z=fa[y];
if(isnrt(y)) (ch[z][1]==y)^(ch[y][1]==x)?rotate(x):rotate(y);
rotate(x);
}
}

变化很容易看出:当前辅助树的根需要特判(代码中用 \(\texttt{isnrt(x)}\) 来实现)。具体的不再解释了。同时注意 \(\texttt{pushdown}\) .

4. 模板

模板题:给你一棵树,每个节点上一个权值,支持加边删边,修改单点权值,查询路径 \(\texttt{xor}\) 和 .

代码:略。

关于 LCT 的具体用途和题目,在(2)中详细讲解。

LCT(1)的更多相关文章

  1. 一堆LCT板子

    搞了一上午LCT,真是累死了-- 以前总觉得LCT高大上不好学不好打,今天打了几遍感觉还可以嘛= =反正现在的水平应付不太难的LCT题也够用了,就这样好了,接下来专心搞网络流. 话说以前一直YY不出来 ...

  2. 动态树之LCT(link-cut tree)讲解

    动态树是一类要求维护森林的连通性的题的总称,这类问题要求维护某个点到根的某些数据,支持树的切分,合并,以及对子树的某些操作.其中解决这一问题的某些简化版(不包括对子树的操作)的基础数据结构就是LCT( ...

  3. 在此为LCT开一个永久的坑

    其实我连splay都还不怎么会. 今天先抄了黄学长的bzoj2049,以后一定要把它理解了. 写LCT怎么能不%数据结构大神yeweining呢?%%%chrysanthemums  %%%切掉大森林 ...

  4. 【BZOJ2157】旅游 LCT

    模板T,SB的DMoon..其实样例也是中国好样例...一开始不会复制,yangyang:找到“sample input”按住shift,按page down.... #include <ios ...

  5. 【BZOJ3669】[Noi2014]魔法森林 LCT

    终于不是裸的LCT了...然而一开始一眼看上去这是kruskal..不对,题目要求1->n的路径上的每个点的两个最大权值和最小,这样便可以用LCT来维护一个最小生成路(瞎编的...),先以a为关 ...

  6. 【BZOJ1180】: [CROATIAN2009]OTOCI & 2843: 极地旅行社 LCT

    竟然卡了我....忘记在push_down先下传父亲的信息了....还有splay里for():卡了我10min,但是双倍经验还是挺爽的,什么都不用改. 感觉做的全是模板题,太水啦,不能这么水了... ...

  7. 【BZOJ3282】Tree LCT

    1A爽,感觉又对指针重怀信心了呢= =,模板题,注意单点修改时splay就好,其实按吾本意是没写的也A了,不过应该加上能更好维护平衡性. ..还是得加上好= = #include <iostre ...

  8. BZOJ2888 资源运输(LCT启发式合并)

    这道题目太神啦! 我们考虑他的每一次合并操作,为了维护两棵树合并后树的重心,我们只好一个一个的把节点加进去.那么这样一来看上去似乎就是一次操作O(nlogn),但是我们拥有数据结构的合并利器--启发式 ...

  9. LCT裸题泛做

    ①洞穴勘测 bzoj2049 题意:由若干个操作,每次加入/删除两点间的一条边,询问某两点是否连通.保证任意时刻图都是一个森林.(两点之间至多只有一条路径) 这就是个link+cut+find roo ...

  10. 链剖&LCT总结

    在搞LCT之前,我们不妨再看看喜闻乐见的树链剖分. 树链剖分有一道喜闻乐见的例题:NOI2015 软件包管理器 如果你看懂题目了,你就会明白它是叫你维护一个树,这棵树是不会动的,要兹磁子树求和,子树修 ...

随机推荐

  1. 【快学SpringBoot】过滤XSS脚本攻击(包括json格式)

    若图片查看异常,请前往掘金查看:https://juejin.im/post/5d079e555188251ad81a28d9 XSS攻击是什么 XSS攻击全称跨站脚本攻击,是为不和层叠样式表(Cas ...

  2. Immediate Decodability[UVA644](Trie入门)

    传送门 题意:给出一些数字串,判断是否有一个数字串是另一个串的前缀. 这题真的可以算是Trie树的一道模板题了. 先把Trie树建好,建树的时候记录一个sum表示一个节点有多少个串会包含此节点,然后再 ...

  3. java 获取(格式化)日期格式

    // 参考: https://www.cnblogs.com/blog5277/p/6407463.htmlpublic class DateTest { // 支持时分秒 private stati ...

  4. Memcache和Redis的详细理解与区别

    1. Memcache Memcache是一个高性能,分布式内存对象缓存系统,通过在内存中缓存一个巨大的hash表,他能够存储包括图像,文件,索引,sql语句结果等数据,可以理解为它理解为一个为提升读 ...

  5. java的基本类型和对应的封装类

    封装,是java这门语言的重要核心思想之一,封装也是对面向对象这一思想很好的体现. 在很多情况下,我们需要对数据进行一些转换,如:将一字符串"123"转换成int类型的123,或者 ...

  6. Day4-A-最短路 HDU2544

    在每年的校赛里,所有进入决赛的同学都会获得一件很漂亮的t-shirt.但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以现在他们想要寻找最短的从商店到赛场的路线,你可以帮助 ...

  7. Day3-D-Protecting the Flowers POJ3262

    Farmer John went to cut some wood and left N (2 ≤ N ≤ 100,000) cows eating the grass, as usual. When ...

  8. 073、Java面向对象之利用构造方法为属性赋值

    01.代码如下: package TIANPAN; class Book { // 定义一个新的类 private String title; // 书的名字 private double price ...

  9. 超大数据量操作 java程序优化[转载]

        一个表中有1000万以上的数据,要对其进行10万次以上的增删查改的操作,请问如何优化java程序对数据库的操作? 通过使用一些辅助性工具来找到程序中的瓶颈,然后就可以对瓶颈部分的代码进行优化. ...

  10. 洛谷 P4287 [SHOI2011]双倍回文题解

    前言 用了一种很奇怪的方法来解,即二分判断回文,再进行某些奇怪的优化.因为这个方法很奇怪,所以希望如果有问题能够 hack 一下. 题解 我们发现,这题中要求的是字符串 \(SS'SS'\),其中 \ ...