树上数据结构——LCT
树上数据结构——LCT
概述
LCT是一种强力的树上数据结构,支持以下操作:
- 链上求和
- 链上求最值
- 链上修改
- 子树修改
- 子树求和
- 换根
- 断开树上一条边
- 连接两个点,保证连接后仍然是一棵树。
基本概念
LCT是对树的实链剖分,即把所有边划分为实边和虚边
类似于重链剖分,每个点连向子节点中的实链至多只会有一条,把这条实边连向的儿子叫做实儿子
把一些实边连接的点构成的链叫做实链,容易发现实链之间没有共同点
需要注意的是一个不在实边上的点(一些叶节点)也视为一条没有实边的实链
于是实链之间一定是用虚边链接的
要涉及动态删连边操作,于是使用splay来维护一条实链,splay是LCT的辅助树
此处splay的深度按中序遍历严格递增
由于用splay维护,LCT的实边是动态的,可以改变
核心操作
access(x):让x到根节点的所有边均为实边,并且x没有实儿子
这个推荐flash_hu的博客,简单易懂
稍微说一下,每次操作先把当前要连的点splay到当前splay的根,由于splay中深度按中序遍历递增,此时根的右儿子一定是之前连的实链,需要去掉
于是把之前的点连到当前根的右儿子就行了
注意此时一些\(fa,son,isroot\)之类的信息改变了,需要\(push\)_\(up\)
void access(int x){
for(int y=0;x;y=x,x=fa[x]){ //y是之前的根,x是当前需要连的点
splay(x); ch[x][1]=y;
push_up(x);
}
}
其他操作
makeroot
换根操作
access(x)之后x是深度最大的点
所以splay(x)之后,x在splay中一定没有右子树,这个时候翻转整个splay,所有点的深度就都倒过来了,x成为深度最小的点,即为根节点
void pushr(int x){
swap(ch[x][0],ch[x][1]);
r[x]^=1;
}
void makeroot(int x){
access(x); splay(x);
pushr(x);
}
findroot
找所在树的树根,可以用来判断两点之间的连通性(两点所在树相同则有唯一相同根
int findroot(int x){
access(x);splay(x);
while(c[x][0]) push_down(x),x=ch[x][0];//寻找深度最小的点,此处push_down是为了x到跟的标记放完,好判连通性
splay(x);//多多splay有益健康
return 0;
}
split
把一条路径拉成一个splay
void spilt(int x,int y){
makeroot(x);access(y);
splay(y);
}
link
连一条边,保证连完还是一棵树
不保证合法:
int link(int x,int y){
makeroot(x);
if(findroot(y)==x) return 0;
fa[x]=y; //把x作为y的儿子
return 1;
}
保证合法:
void link(int x,int y){
makeroot(x);
fa[x]=y;
}
此处连的边是虚边(感受到实链剖分的方便了罢
cut
断边
保证存在:
void cut(int x,int y){
split(x,y);
fa[x]=ch[y][0]=0; }
不存在此边的时候是什么情况呢?
先把x给\(makeroot\)到根
x和y不连通 (\(findroot\))
在同一splay中而没有直接连边 (\(f[y]==x\)且\(!c[y][0]\))
(考虑其他的点在哪里,在findroot之后x到了根节点,如果x和y之间有点,只能是在y到根的路径上或者y的左儿子上)
int cut(int x,int y){
makeroot(x);
if(findroot(y)!=x||fa[y]!=x||ch[y][0]) return 0;
fa[y]=ch[x][1]=0;
push_up(x);
return 1;
}
nroot
naiive的操作,判断此点是否不是当前splay的根节点
int nroot(int x){
return (ch[fa[x]][1]==x||ch[fa[x]][0]==x);
}
splay 的特殊性
此处splay的标记一定要从上往下放,也就是先开个栈把标记放完再旋转
完整模板
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
using namespace std;
#define reg register int
#define il inline
#define ls ch[x][0]
#define rs ch[x][1]
int read(){
int x=0,pos=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') pos=0;
for(;isdigit(ch);ch=getchar()) x=(x<<1)+(x<<3)+ch-'0';
return pos?x:-x;
}
const int N = 400025;
int fa[N],ch[N][2],v[N],s[N],st[N],r[N];
il int nroot(int x){
return ch[fa[x]][1]==x||ch[fa[x]][0]==x;
}
il int get(int x){
return ch[fa[x]][1]==x;
}
il void push_up(int x){
s[x]=v[x]^s[ls]^s[rs];
}
il void pushr(int x){
swap(ls,rs);r[x]^=1;
}
il void push_down(int x){
if(r[x]){
if(ls) pushr(ls);
if(rs) pushr(rs);
r[x]=0;
}
}
il void rotate(int x){
int f=fa[x],g=fa[f],kx=get(x),kf=get(f);
if(nroot(f)) ch[g][kf]=x;
fa[x]=g;
/*if(ch[x][kx^1])*/ fa[ch[x][kx^1]]=f;
ch[f][kx]=ch[x][kx^1];
fa[f]=x;ch[x][kx^1]=f;
push_up(f);push_up(x);
}
il void push(int x){
if(nroot(x)) push(fa[x]);
push_down(x);
}
il void splay(int x){
int f,g;
push(x);//fuctional stack
while(nroot(x)){
f=fa[x],g=fa[f];
if(nroot(f)){
rotate(get(x)==get(f)?f:x);
}
rotate(x);
}
}
il void access(int x){
for(reg y=0;x;y=x,x=fa[x]){
splay(x);rs=y;push_up(x);
}
}
il void makeroot(int x){
access(x);splay(x);pushr(x);
}
il void spilt(int x,int y){
makeroot(x);
access(y);splay(y);
}
il int findroot(int x){
access(x);splay(x);
while(ls) x=ls;
splay(x);
return x;
}
il void link(int x,int y){
makeroot(x);
if(findroot(y)!=x) fa[x]=y;
}
il void cut(int x,int y){
makeroot(x);
if(findroot(y)==x&&fa[y]==x&&!ch[y][0]){
fa[y]=ch[x][1]=0;
push_up(x);
}
}
int n,m;
int main(){
n=read();m=read();
for(reg i=1;i<=n;i++){
v[i]=read();
}
for(reg i=1;i<=m;++i){
int opt=read(),x=read(),y=read();
if(opt==0){
spilt(x,y);printf("%d\n",s[y]);
}else if(opt==1){
link(x,y);
}else if(opt==2) cut(x,y);
else splay(x),v[x]=y;
}
return 0;
}
之后可能会补自己做的LCT题(咕
在创作本文的过程中,参考了以下文章:
- [flash_hu大佬的博客][https://www.cnblogs.com/flashhu/p/8324551.html]
- [NOI级别的超强数据结构——Link-cut-tree(动态树)学习小记][https://blog.csdn.net/qq_36551189/article/details/79152612]
- 成都七中的LCT课件(有一点吧
- Yang Zhe 2007年的论文
树上数据结构——LCT的更多相关文章
- 2019暑期金华集训 Day5 树上数据结构
自闭集训 Day5 树上数据结构 前置知识 点分治 边分治 树链剖分 LCT Top Tree LCT时间复杂度 线段树每次查询是严格\(\log n\)的,然而splay维护连续段的时候,如果每次查 ...
- 模板—数据结构—LCT
模板—数据结构—LCT Code: #include <cstdio> #include <algorithm> using namespace std; #define N ...
- ZROI 19.08.01 树上数据结构
1.总览 LCT 链分治(树剖) 点/边分治 2.点分治 一棵树,点有\(0/1\),多次修改,询问最远的两个\(1\)距离. 建出点分树,每个子树用堆维护:①最远的\(1\)距离:②它的每个儿子的① ...
- ZROI 暑期高端峰会 A班 Day4 树上数据结构
FBI Warning:本文含有大量人类的本质之一. 你经历过绝望吗? [ZJOI2007]捉迷藏 询问树上最远黑点对. 动态边分治可以比点分治少一个 \(\log\). bzoj3730 咕了. [ ...
- CF1111E Tree 动态规划+LCT
这个题的思路非常好啊. 我们可以把 $k$ 个点拿出来,那么就是求将 $k$ 个点划分成不大于 $m$ 个集合的方案数. 令 $f[i][j]$ 表示将前 $i$ 个点划分到 $j$ 个集合中的方案数 ...
- OI题目类型总结整理
## 本蒟蒻的小整理qwq--持续更新(咕咕咕) 数据结构 数据结构 知识点梳理 数据结构--线段树 推荐yyb dalao的总结--戳我 以后维护线段树还是把l,r写到struct里面吧,也别写le ...
- zhengrui集训D1-D5笔记
Day_1 计数 它咕掉了 Day_1 序列数据结构 它咕掉了 Day_2 线性代数 高斯消元\Large{高斯消元}高斯消元 普通版:略 模质数:求逆 模合数:exgcd 逆矩阵\Large{逆矩阵 ...
- 【BZOJ-2555】SubString 后缀自动机 + LinkCutTree
2555: SubString Time Limit: 30 Sec Memory Limit: 512 MBSubmit: 1936 Solved: 551[Submit][Status][Di ...
- The Last Week
二轮省选前的最后一周了呢. 一路走到这里,真的很希望能继续走下去. 好好调整一下状态,争取能有机会买D吧(虽然现在似乎D也没什么用了 day1 多项式 多项式ln 多项式exp day2 数据结构 L ...
随机推荐
- Spring框架核心知识介绍
一:spring框架介绍 1.spring框架是为了解决复杂的企业级应用而创建的, 使用Spring可以让简单的JavaBean实现之前只有EJB才能完成的事情.但是Spring不仅仅局限于服务器 ...
- (一)spring aop的两种配置方式。
sring aop的方式有两种:(1)xml文件配置方式(2)注解的方式实现,我们可以先通过一个demo认识spring aop的实现,然后再对其进行详细的解释. 一.基于注解的springAop配置 ...
- E-MAZE_2019牛客暑期多校训练营(第二场)
题意 给出n行m列的迷宫0可走1不可走,有两个操作,操作1变换点(a,b)的值,操作2查询(1,a)到(n,b)的方案数 题解 设\(F[i][j]\)为第i-1行到达第i行第j列的方案数,若点\(( ...
- HDU4614Vases and Flowers 二分+线段树;
参考:https://blog.csdn.net/ophunter_lcm/article/details/9879495 题意: 有n个花瓶,有两种操作,1.从a开始放b朵花,有花的花瓶跳过,2 ...
- JS-特效 ~ 02. 屏幕滚动事件、 DTD、scroll、顶部悬浮导航、两侧跟随广告、返回顶部小火煎
ceil 向上取整 floor 向下取整 round 四舍五入 缓动动画 动画原理 = 盒子位置 + 步长 清除定时器 步长越来越小 盒子位置 = 盒子本身位置 + (目标位置-本身位置)/n(最好为 ...
- MATLAB之基本语法
常用命令 dir:列出当前目录下的所有文件 clc:清除命令窗 clear all:清除环境(从内存中清除所有变量) who:将内存中的当前变量以简单形式列出 close all: 关闭所有的 Fig ...
- Redis真的那么好用吗
Redis是什么 Redis是一个开源的底层使用C语言编写的key-value存储数据库.可用于缓存.事件发布订阅.高速队列等场景.而且支持丰富的数据类型:string(字符串).hash(哈希).l ...
- 六星教育php vip视频(分享)
最近看的一个swoole的课程,应该也算是vip课程了,不是公开的直播课 比较有特点有一定深度,swoole的实战教程一直也不多,结合swoole构建一个新型框架,最后讲解如何实现分布式RPC的调用. ...
- 算法与数据结构基础 - 递归(Recursion)
递归基础 递归(Recursion)是常见常用的算法,是DFS.分治法.回溯.二叉树遍历等方法的基础,典型的应用递归的问题有求阶乘.汉诺塔.斐波那契数列等,可视化过程. 应用递归算法一般分三步,一是定 ...
- WAP自助建站平台娃派宣布关闭 感谢建站之路有你的启蒙
如题所示的这篇文章是我心血来潮在网上搜索到的,写的挺让我感同身受的,不妨先看一下原文吧. 原文 不知是偶然还是"冥冥定数",最后一次访问娃派建站(wap.ai)已有数十月之久了,突 ...