首页 > 编程语言 >算法学习笔记(16):Link Cut Tree

算法学习笔记(16):Link Cut Tree

时间:2024-05-07 09:04:36浏览次数:17  
标签:Cut rs int 16 Tree add ch ls mod

简称LCT(不是Li Chao Tree), 是一种非常强大的数据结构。

声明

该博客写来很大部分目的是帮助自己理解, 笔者水平有限, 没办法完全原创, 有很多内容源自于OI-wiki,和网上博客, 见谅。

功能

考虑一些问题:

  1. 树上单点查, 树上路径修改, 这是树上差分可以解决的。
  2. 那么如果路径查, 路径修呢? 或者再加上子树查, 子树修改, 单点查呢? 树链剖分可以解决。
  3. 那么再加上可以任意删边连边呢?(保证一直都是一棵树), 这时候就需要LCT了。

所以LCT可以解决:

  1. 修改两点间路径权值。
  2. 查询两点间路径权值和。
  3. 修改某点子树权值。
  4. 查询某点子树权值和。
  5. 断边连边。
  6. 还可以维护一些奇怪信息, 这是因为Splay可以搞区间操作。

算法

实链剖分

这是比较类似重链剖分的, 重链剖分的重链是根据子树的大小来确定的, 而我们的实链是自己钦定的, 随时可以变化。
所以原树会被我们剖成若干个实链, 每条实链由一颗Splay维护。 并且这个Splay是根据深度作为BST的键值的。
所以, 就会有一些美好性质:

  1. 辅助树由多棵 Splay 组成,每棵 Splay 维护原树中的一条路径,且中序遍历这棵 Splay 得到的点序列,从前到后对应原树「从上到下」的一条路径。
  2. 原树每个节点与辅助树的 Splay 节点一一对应。
  3. 辅助树的各棵 Splay 之间并不是独立的。每棵 Splay 的根节点的父亲节点本应是空,但在 LCT 中每棵 Splay 的根节点的父亲节点指向原树中 这条链 的父亲节点(即链最顶端的点的父亲节点)。这类父亲链接与通常 Splay 的父亲链接区别在于儿子认父亲,而父亲不认儿子,对应原树的一条 虚边。因此,每个连通块恰好有一个点的父亲节点为空。
  4. 由于辅助树的以上性质,我们维护任何操作都不需要维护原树,辅助树可以在任何情况下拿出一个唯一的原树,我们只需要维护辅助树即可。

操作

考虑我们访问树上路径, 这就需要两个很重要的操作。
Access()

void access(int u) {
	for (int f = 0; u; f = u, u = fa[u]) 
		splay(u), ch[u][1] = f, maintain(u);
}

表示将节点 \(x\) 到根的路径全部钦定为实边, 这样就可以得到 \(x\) 到根的路径了。 因为只有实边才会被Splay维护, 才可以进行操作。
考虑实现过程, 就是把 \(x\) 翻到所在Splay的根, 然后把他父亲的右儿子指向它(这一步相当于就是把这条虚边钦定为实边), 一直到根, 就OK了。
Makeroot()

void makeroot(int u) {
	access(u);
	splay(u);
	swap(ls, rs);
	rev[u] ^= 1;
}

考虑access()操作只能访问 \(x\) 到根的路径, 但实际上树上路径不一定深度严格递增, 解决办法就是把某一个点钦定为根, 考虑一个性质, 如果把一个树看成一个内向树或者外向树, 那么换根就是把 \(x\) 到原来的根的路径上的边全部反向, 所以我们用Splay的区间翻转, 就可以实现换根。
实现过程: 访问 \(x\) 到根节点的路径, 也就是 access(x), 然后将这条路径的Splay的节点 \(x\) 翻到根, 然后打翻转标记就行了。

其他操作就很简单啦, 建议自行yy以加强对这两个核心操作的理解。

模板题

P1501 [国家集训队] Tree II

点击查看代码
#include<bits/stdc++.h>
using namespace std;
#define int long long
const int N = 1e5 + 10;
const int mod = 51061;
int n, q;
char op;
struct Splay{
	int ch[N][2], fa[N], siz[N], val[N], sum[N], rev[N], add[N], mul[N];
	#define ls ch[u][0]
	#define rs ch[u][1]
	void clear(int u) {
		ls = rs = fa[u] = siz[u] = val[u] = sum[u] = rev[u] = add[u] = 0;
		mul[u] = 1;
	}
	int get(int u) { return (ch[fa[u]][1] == u); }
	int isroot(int u) {
		clear(0); 
		return ch[fa[u]][0] != u && ch[fa[u]][1] != u;
	} 
	void maintain(int u) {
		clear(0);
		siz[u] = (siz[ls] + 1 + siz[rs]) % mod;
		sum[u] = (sum[ls] + val[u] + sum[rs]) % mod;
	}
	void pushdown(int u) {
		clear(0);
		if (mul[u] != 1) {
			if (ls)
				mul[ls] = (mul[u] * mul[ls]) % mod,
				val[ls] = (mul[u] * val[ls]) % mod,
				sum[ls] = (mul[u] * sum[ls]) % mod,
				add[ls] = (mul[u] * add[ls]) % mod;
			if (rs) 
				mul[rs] = (mul[u] * mul[rs]) % mod,
				val[rs] = (mul[u] * val[rs]) % mod,
				sum[rs] = (mul[u] * sum[rs]) % mod,
				add[rs] = (mul[u] * add[rs]) % mod;
			mul[u] = 1;
		}
		if (add[u]) {
			if (ls) 
				add[ls] = (add[u] + add[ls]) % mod,
				val[ls] = (add[u] + val[ls]) % mod,
				sum[ls] = (add[u] * siz[ls] % mod + sum[ls]) % mod;
			if (rs)
				add[rs] = (add[u] + add[rs]) % mod,
				val[rs] = (add[u] + val[rs]) % mod,
				sum[rs] = (add[u] * siz[rs] % mod + sum[rs]) % mod;
			add[u] = 0;
		}
		if (rev[u]) {
			if (ls) rev[ls] ^= 1, swap(ch[ls][0], ch[ls][1]);
			if (rs) rev[rs] ^= 1, swap(ch[rs][0], ch[rs][1]);
			rev[u] = 0;
		}
	}
	void update(int u) {
		if (!isroot(u)) update(fa[u]);
		pushdown(u);
	}
	void rotate(int u) {
		int f = fa[u], gf = fa[f], chk = get(u);
		fa[u] = gf;
		if (!isroot(f)) ch[gf][f == ch[gf][1]] = u;
		ch[f][chk] = ch[u][chk ^ 1];
		if (ch[u][chk ^ 1]) fa[ch[u][chk ^ 1]] = f;
		ch[u][chk ^ 1] = f;
		fa[f] = u;
		maintain(f); 
		maintain(u);  
		maintain(gf);
	}
	void splay(int u) {
		update(u);
		for (int f = fa[u]; f = fa[u], !isroot(u); rotate(u)) {
			if (!isroot(f)) rotate(get(u) == get(f) ? f : u);
		}
	}
	void access(int u) {
		for (int f = 0; u; f = u, u = fa[u]) 
			splay(u), ch[u][1] = f, maintain(u);
	}
	void makeroot(int u) {
		access(u);
		splay(u);
		swap(ls, rs);
		rev[u] ^= 1;
	}
	int find(int u) {
		access(u); 
		splay(u);
		while (ls) u = ls;
		splay(u);
		return u;
	}
	void add_edge(int u, int v) {
		if (find(u) != find(v)) 
			makeroot(u), fa[u] = v; 
	}
	void modify_add(int u, int v, int z) {
		makeroot(u); access(v); splay(v);
		val[v] = (val[v] + z) % mod;
		sum[v] = (sum[v] + z * siz[v] % mod) % mod;
		add[v] = (add[v] + z) % mod;
	}
	void modify_mul(int u, int v, int z) {
		makeroot(u); access(v); splay(v);
		val[v] = val[v] * z % mod;
		sum[v] = sum[v] * z % mod;
		mul[v] = mul[v] * z % mod;
	}
	void cut(int u, int v) { 
		makeroot(u); access(v); splay(v); 
		if (ch[v][0] == u && !rs)
			ch[v][0] = fa[u] = 0; 
	}
	void link(int u, int v) { 
		makeroot(u); splay(u); 
		if (find(u) != find(v)) fa[u] = v; 
	}
	int query(int u, int v) { 
		makeroot(u); access(v); splay(v); 
		return sum[v]; 
	}
}T; 
signed main() {
	scanf("%lld%lld", &n, &q);
	for (int i = 1; i <= n; i++) 
		T.val[i] = 1, T.maintain(i);
	for (int i = 1, u, v; i < n; i++) 
		scanf("%lld%lld", &u, &v), T.add_edge(u, v);
	for (int i = 1, op, u, v, z; i <= q; i++) {
		scanf(" %c%lld%lld", &op, &u, &v);
		switch(op) {
			case '+': scanf("%lld", &z); T.modify_add(u, v, z); break;
			case '-': T.cut(u, v); scanf("%lld%lld", &u, &v); T.link(u, v); break;
			case '*': scanf("%lld", &z); T.modify_mul(u, v, z); break;
			case '/': printf("%lld\n", T.query(u, v)); break; 
		}
	}
	return 0;
}

标签:Cut,rs,int,16,Tree,add,ch,ls,mod
From: https://www.cnblogs.com/qerrj/p/18176446

相关文章

  • 『Solution』Codeforces 372D Choosing Subtree is Fun
    首先这个\(k\)的限制不是很好入手,考虑先从如果选取了区间\([l,r]\)来入手。那么此时连通块最小的\(siz\)就是把这些点拎出来建虚树对应在原树上的所有点。那么这个有个结论,考虑按\(\operatorname{dfn}\)序排序后的点为\(p_0\simp_{k-1}\),那么对应的最小\(sz\)就......
  • MySQL夺命16问,你能坚持到第几问(转)
    原文:https://zhuanlan.zhihu.com/p/5344154091、数据库三大范式是什么?**第一范式:每个列都不可以再拆分。第二范式:在第一范式的基础上,非主键列完全依赖于主键,而不能是依赖于主键的一部分。第三范式:在第二范式的基础上,非主键列只依赖于主键,不依赖于其他非主键。在设计数据库结......
  • Unable to execute SonarScanner analysis: Fail to get bootstrap index from server
    1.背景编辑gitlab-ci流水线时,代码分析的job,maven使用sonar报错-mvncleanverifysonar:sonar-Dsonar.login=30c55d3b8d3d2569431fb39f3c488c90643a68442.错误信息[ERROR]Failedtoexecutegoalorg.sonarsource.scanner.maven:sonar-maven-plugin:3.11.0.3922:sonar(def......
  • openGauss btree-索引故障情况下应对策略
    btree索引故障情况下应对策略问题现象偶发索引丢失错误,报错如下。ERROR:index'xxxx_index'containsunexpectedzeropage或ERROR:index'pg_xxxx_index'containsunexpectedzeropage或ERROR:compresseddataiscorrupt原因分析该类错误是因为索引发生故障导......
  • snmp trap的162端口down解决方法
    电脑是WIN10的系统,进行snmptrap的测试发现,162端口down掉了1.cmd打开命令提示符窗口查看UDP端口占用用命令netstat-ano,发现没有162的端口被占用2.service里没有看到snmp安装1)打开设置中的[开发人员模式],设置->更新与安全->开发者选项->开发人员模式2)添加SNMP,设置->应用->应......
  • window系统启动snmptraps服务162端口
    在Windows系统中启动snmptraps服务通常意味着你需要启用SNMP服务并配置它来监听162端口,该端口是SNMPtrap消息的默认端口。以下是如何在Windows上配置snmptraps服务的步骤:打开“控制面板”。选择“程序和功能”,然后点击“打开或关闭Windows功能”。在弹出的“Windows功能”对......
  • BCM53161XUB0KLFBG、BCM53161XMB0KLFBG、BCM53161XMB0ILFBG: 超低功耗2.5GE交换机介绍
    产品介绍BCM5316X超低功耗2.5GE交换机设计用于SMB、工业和服务提供商市场中的多GE应用。BCM5316X交换机支持四个2.5GESGMII+端口、两个2.5GE/10GEXFI/SFI端口以及多达八个带集成GPHY的10/100/1000Base-T端口。BCM5316X交换机采用28nmRoboSwitch™架构(也称为Robo-2)。BCM5316X集......
  • postgresql 16 主备/主从
    注意事项基于postgresql16版本相关核心概念介绍预写日志机制(WAL)数据持久化是指提交事务后对系统的影响是永久的,即使数据库重启或崩溃,数据不会丢失。最简单的做法是事务提交后,数据就立刻持久化到磁盘中。但是内存和磁盘之间的IO操作是最影响性能的,所以会将一部分数据保......
  • LeetCode 1373. Maximum Sum BST in Binary Tree
    原题链接在这里:https://leetcode.com/problems/maximum-sum-bst-in-binary-tree/description/题目:Givena binarytree root,return themaximumsumofallkeysof any sub-treewhichisalsoaBinarySearchTree(BST).AssumeaBSTisdefinedasfollows:Thel......
  • CF1967C. Fenwick Tree-算子展开,树状数组的结构
    link:https://codeforces.com/problemset/problem/1967/C题意:定义\(f(a)=s\)中的\(f\)表示把序列\(a\)映射为其树状数组的操作(\(s\)就是对应的树状数组),并且操作是在取模下作的,已知\(f^k(a)=b\),已知序列\(b\)和自然数\(k\),求\(a\).\(1\leqn\leq2\times10^5,1\leq......