LCT

今天,YCC 本来是想搞一搞数论,但是老吕说:今天我闲着,给你们讲一讲LCT。

可是,我不闲啊!

LCT 是什么?

是一个由若干棵子结点无序的有根树组成的森林,支持对树的分割, 合并, 对某个点到它的根的路径的某些操作, 以及对某个点的子树进行的某些操作。

基本概念

原树:就是原来的树,实孩子,实边,实链,都是原树中的。

辅助树:是很多splay 构成的树,程序中真正操作的都是辅助树。

实孩子:这个实孩子与轻重链剖分中的重孩子还不同。

轻重链剖分中的重孩子是根据子树的大小来确定的,但是LCT中的实孩子是不确定的,当我们需要它时,就将它从虚孩子变为实孩子。

实边:连接实孩子的边称为实边。

实链:实边组成的的链为实链。

基本思想

思想和树链剖分的思想类似,每个点(叶子点除外)中最多有一个孩子为实孩子(可能没有)。连接实孩子的边称为实边,实边组成的的链为实链。

把实链交给splay维护。按深度作为关键字,每一个 splay 的中序遍历就是在原树中从上往下的一条链。

这样还剩下一些虚孩子,和各个分散的 splay。 让每棵 splay 的根节点的父亲节点指向原树中这条链 的父亲节点(即实链链顶的父亲节点)。

这类连边与通常 Splay 中(也就是原树上的实边)的连边有所不同,区别在于这儿子认父亲,而父亲不认儿子,对应着原树的一条 虚边

基础操作

首先是 splay 和 rotate。

void splay(int p) 
{
	Push_down(p);
	for(; !isroot(p); rotate(p)) 
		if(!isroot(fa(p))) rotate(get(fa(p)) == get(p) ? fa(p) : p);
}
void rotate(int p) 
{
	int fath = fa(p), x = get(p);
	if(!isroot(fa(p))) t[fa(fa(p))].son[get(fa(p))] = p;
	fa(p) = fa(fa(p));
	t[fath].son[x] = t[p].son[x ^ 1];
	fa(t[p].son[x ^ 1]) = fath;
	t[p].son[x ^ 1] = fath;
	fa(fath) = p;
	push_up(fath), push_up(p);
}

isroot(x):判断x是否是辅助树的根。

方法:splay 的根结点的父亲并不认这个孩子。原树的根所在的 splay 的父亲点是 \(0\)。

bool isroot(int x)//判断是否为splay中的根 
{
	return ls(fa(x)) != x && rs(fa(x)) != x;//根节点的父亲不认它这个孩子 
}

access(x):把 \(x\) 点下面的实边断开,并把 \(x\)​ 点一路向上边到树的根,(也就是,打通根节点到指定节点的实链,使得一条中序遍历以根开始、以指定点结束的 splay出现。)

方法:把 \(x\) 点伸展到 splay 的根,再把它的右子树连到 y,也就是连上了下一层的实链,然后 y 更新为x,而 x 更新为 x 的父亲,继续向上连接。由于 y 的初始值为0,所以第一次连接也就是断开了下面的实边。

for(int x = 0; p; x = p, p = fa(p)) 
    splay(p), rs(p) = x, push_up(p);

link(x,y):连接两个点。

方法:把 \(x\) 点变成所在原树的根,然后把 \(x\) 点的父亲变成 \(y\) 。

void link(int x, int y) 
{
	makeroot(x);
	if(find(y) != x) fa(x) = y;
}

cut(x,y):断开两个点间的边。

void cut(int x, int y) 
{
	makeroot(x);
	if(find(y) != x || fa(y) != x || ls(y)) return ;
	fa(y) = rs(x) = 0;
	push_up(x);
}

makeroot(x):把x点变为树的根。

方法:首先的把 \(x\) 点 access 到根,把 \(x\) 点到根变成一个 splay,然后把 \(x\) 伸展到根。由于 \(x\) 点是辅助树在原树中最下面的点,所以这时其它的点都在 \(x\) 的左子树上,只要把左子树变成右子树,\(x\) 也就变成了根。

void makeroot(int p) 
{
	access(p); splay(p); f(p);
}

find(x):查找 \(x\)​​​​ 所在树的根。

方法:首先把 \(x\) 点 access 到原树的根,并把它 splay 到辅助树的根,这时原树的根就是 \(x\) 左子树中最左侧的点。

int find(int p) 
{
	access(p); splay(p);
	while(ls(p)) push_down(p), p = ls(p);
	splay(p);
	return p;
}

/

/*
Date:2021.9.7
Source:luogu 3690
konwledge:
*/
#include <iostream>
#include <cstdio>
#define orz cout << "AK IOI" << "\n"
#define fa(x) t[x].fa
#define ls(x) t[x].son[0]
#define rs(x) t[x].son[1]
#define get(x) (rs(fa(x)) == x)

using namespace std;
const int maxn = 3e5 + 10; 

inline int read()
{
	int f = 0, x = 0; char ch = getchar();
	while(!isdigit(ch)) f |= (ch == '-'), ch = getchar();
	while(isdigit(ch)) x = x * 10 + (ch ^ 48), ch = getchar();
	return f ? -x : x;
}
inline void print(int X)
{
	if(X < 0) X = ~(X - 1), putchar('-');
	if(X > 9) print(X / 10);
	putchar(X % 10 + '0');
}

int n, m;
struct node {
	int son[2], fa, val, sum, lzy;
} t[maxn];
int kcnt;
bool isroot(int x)
{
	return ls(fa(x)) != x && rs(fa(x)) != x;
}
void push_up(int p)
{
	t[p].sum = t[ls(p)].sum ^ t[rs(p)].sum ^ t[p].val;
}
void f(int p) 
{
	swap(ls(p), rs(p));
	t[p].lzy ^= 1;
}
void push_down(int p) 
{
	if(t[p].lzy) f(ls(p)), f(rs(p)), t[p].lzy = 0;
}
void rotate(int p) 
{
	int fath = fa(p), x = get(p);
	if(!isroot(fa(p))) t[fa(fa(p))].son[get(fa(p))] = p;
	fa(p) = fa(fa(p));
	t[fath].son[x] = t[p].son[x ^ 1];
	fa(t[p].son[x ^ 1]) = fath;
	t[p].son[x ^ 1] = fath;
	fa(fath) = p;
	push_up(fath), push_up(p);
}
void Push_down(int p) 
{
	if(!isroot(p)) Push_down(fa(p));
	push_down(p);
}
void splay(int p) 
{
	Push_down(p);
	for(; !isroot(p); rotate(p)) 
		if(!isroot(fa(p))) rotate(get(fa(p)) == get(p) ? fa(p) : p);
}
void access(int p) 
{
	for(int x = 0; p; x = p, p = fa(p)) splay(p), rs(p) = x, push_up(p);
}
void makeroot(int p) 
{
	access(p); splay(p); f(p);
}
int find(int p) 
{
	access(p); splay(p);
	while(ls(p)) push_down(p), p = ls(p);
	splay(p);
	return p;
}
void split(int x, int y) 
{
	makeroot(x); access(y); splay(y);
}
void link(int x, int y) 
{
	makeroot(x);
	if(find(y) != x) fa(x) = y;
}
void cut(int x, int y) 
{
	makeroot(x);
	if(find(y) != x || fa(y) != x || ls(y)) return ;
	fa(y) = rs(x) = 0;
	push_up(x);
}

int main() 
{
	n = read(), m = read();
	for(int i = 1; i <= n; i++) t[i].val = read();
	for(int i = 1; i <= m; i++) 
	{
		int opt = read(), x = read(), y = read();
		if(opt == 0) split(x, y), print(t[y].sum), puts("");
		if(opt == 1) link(x, y);
		if(opt == 2) cut(x, y);
		if(opt == 3) splay(x), t[x].val = y;
	}
	return 0;
}
上一篇:Mysql 递归查询


下一篇:[cf1137F]Matches Are Not a Child's Pla