全局平衡二叉树 (GBST) 小记

news/2025/1/15 17:17:31/文章来源:https://www.cnblogs.com/estruct/p/18542699

全局平衡二叉树 (GBST) 小记

以下全局平衡二叉树简称 \(\text{GBST(Globel Balanced Search Tree)}\)

我认识的大多数人,对 \(\text{GBST}\) 的理解基本上都是 静态 \(\text{LCT}\),或者静态 \(\text{Top Tree}\),不过我对 \(\text{LCT}\) 的理解可能还差一点,所以我不打算从阉割 \(\text{LCT}\) 的角度来讲解。

于是下文基本上是从树剖优化的角度讲解。


考虑一个经典的树上问题,\(n\) 个结点,点有点权,\(q\) 次操作,操作有两种:

  • 给定 \(x, y, k\),将 \(x\)\(y\) 路径上的点权值 \(+k\)
  • 给定 \(x, y\),求 \(x\)\(y\) 路径上所有点的权值之和。

首先经典的可以将问题差分,变成前缀链加和前缀链查询。于是下面我们都默认链是树上的前缀,即一条 \(x\) 到根的路径。

对原树重链剖分,注意到无论是修改还是查询,都作用在重链的一段前缀上。这给了我们一些操作的空间。

考虑对于每条重链分别维护线段树,而不是简单的用 \(\text{dfs}\) 序拍平成序列问题,但这样只是常数小一点(真的小了吗?),对于单条重链上做修改/查询的时间复杂度还是 \(O(\log n)\) 的。

我们发现时间复杂度优化不了,本质上的原因的每条重链是独立的。我们考虑用一些数据结构将树剖的 \(\log\) 和线段树的 \(\text{log}\) 联系起来。

具体而言,我们对于重链用一棵二叉搜索树维护。由于二叉搜索树的真实顺序是其中序遍历,画画图就能知道求的是什么。考虑 \(x\)\(x\) 所在重链链顶的贡献,拿出对于每条树边 \(u \rightarrow v\),如果 \(u\)\(v\) 右儿子则将 \(u\) 加入集合 \(s\),特别的 \(x\) 也在 \(s\) 中。会被操作覆盖的点是 \(s\) 中所有点和其左子树的并,这样可以做到一个看似 \(O(1)\) 修改/查询。

但是如果真的建一个完全二叉树,那复杂度又回去了。因为对于一条重链,还是会在该重链对应的二叉树上条 \(O(\log n)\) 步。

于是考虑构造一个类似于 每跳一条边,子树大小翻倍的结构。我们发现轻边已经满足这个要求,要是完全二叉树上的边也满足这个要求,不难想到建二叉树时不取中点为根(取中点就是完全二叉树),而取关于子树大小的带权中点即可。

但是带权中点总会向一边倾斜,所以最坏情况下要条两条边才能使子树大小翻倍。也就是说最坏情况下,\(\text{GBST}\) 上一个点的深度是 \(2 \log n\)。可能又更好的分析/构造方法得到类似 深度不超过 \(\log n + O(1)\) 的结论,不过它已经足够优秀,我就懒得想了。


下面是例题:

代码:

#include <iostream> 
#include <vector>
#include <random>
#include <algorithm>using namespace std;
using uLL = unsigned long long;const int N = 2e5 + 5; int n, q, v, seed;
vector<int> e[N];
int fa[N], siz[N], wson[N];
int lc[N], rc[N], ss[N];
uLL val[N], sum[N], tag[N];
mt19937 R;bool Get_which (int x) {return x == lc[fa[x]] ? 0 : (x == rc[fa[x]] ? 1 : -1);
}int Rand (int l, int r) {return uniform_int_distribution<int>(l, r)(R);
} void Init (int u, int r) {if (r) e[u].erase(find(e[u].begin(), e[u].end(), r));siz[u] = 1; for (auto v : e[u]) {if (v == r) continue;Init(v, u);siz[u] += siz[v];if (siz[v] > siz[wson[u]]) {wson[u] = v; }}
}int Build (vector<int>::iterator itl, vector<int>::iterator itr) {if (itl == itr) return 0; int bot = siz[*prev(itr)], top = siz[*itl] - bot;auto mid = upper_bound(itl, itr, top / 2, [&](int half, int x) -> bool {return siz[x] - bot <= half;});int u = *mid;lc[u] = Build(itl, mid), rc[u] = Build(mid + 1, itr);fa[lc[u]] = fa[rc[u]] = u, ss[u] = ss[lc[u]] + ss[rc[u]] + 1;return u; 
}int Divide (int u) {vector<int> lis;for (int s = u; s; s = wson[s]) {lis.push_back(s);for (auto v : e[s]) {if (v != wson[s])fa[Divide(v)] = s;}}return Build(lis.begin(), lis.end());
}void Node_add (int x, uLL y) {val[x] += y, sum[x] += y * ss[x], tag[x] += y; 
}void Pushdown (int x) {if (tag[x]) {if (lc[x]) Node_add(lc[x], tag[x]);if (rc[x]) Node_add(rc[x], tag[x]);tag[x] = 0; }
}void Pushup (int x) {sum[x] = sum[lc[x]] + sum[rc[x]] + val[x];
}void Add (int x, int y, int whi = -1) {if (fa[x]) {Add(fa[x], y, Get_which(x));}Pushdown(x);if (whi != 0) {val[x] += y, sum[x] += y; if (lc[x]) Node_add(lc[x], y);}if (whi == -1) {for (int i = x; i; i = fa[i]) {Pushup(i);if (Get_which(i) == -1) break;}}
}uLL Query (int x, int whi = -1) {uLL res = 0; if (fa[x]) {res = Query(fa[x], Get_which(x));}Pushdown(x);if (whi != 0) {res += val[x] + sum[lc[x]];}return res;
}int main () {cin.tie(0)->sync_with_stdio(0);cin >> n >> q >> v >> seed, R = mt19937(seed); for (int i = 1, x, y; i < n; ++i) {cin >> x >> y; e[x].push_back(y);e[y].push_back(x);}Init(1, 0), Divide(1); uLL ans = 0; for (int i = 1, o, x, y; i <= q; ++i) {o = Rand(1, 2), x = Rand(1, n); if (o == 1) {y = Rand(1, v);Add(x, y);}else {uLL ret = Query(x);ans ^= i * ret;}}cout << ans << '\n';return 0; 
}

下面是效率比较,以上面的例题为例(\(n = 2 \times 10^5, q = 5 \times 10^6\)):

树的形态 树剖+线段树 \(\text{GBST}\) 暴力跳
随机树(深度为 \(O(\sqrt n)\) \(\text{3.9s}\) \(\text{1.8s}\) \(\text{3.0s}\)
\(\text{1.6s}\) \(\text{2.5s}\) \(\ge \text{100.0s}\)
菊花 \(\text{1.6s}\) \(\text{0.4s}\) \(\text{0.3s}\)
完全二叉树 \(\text{6.8s}\) \(\text{1.5s}\) \(\text{0.6s}\)
链上插点 \(\text{2.1s}\) \(\text{2.3s}\) \(\ge \text{100.0s}\)

这里暴力跳仅作为对照,参考意义不大。为什么 \(\text{GBST}\) 理论复杂度比树剖少个 \(\log n\),实际表现却没有那么优秀?主要原因有以下几个:

  • 树剖本身很难卡满,这里的难卡满指的是重链条数,因为你拍成 \(\text{dfs}\) 序做的话实际上具体查询的区间是什么对时间复杂度影响是不大的。

  • 我写的 \(\text{GBST}\) 没有标记永久化,那么每次全部 \(\text{pushdown}\) 一遍常数其实是挺大的。

并且 \(\text{GBST}\) 还有一个缺陷。前面我们讨论的都是默认差分后的情况,但任意路径的前缀修改/求和需要差分成四条前缀链,但树剖可以自然的处理掉,所以如果真要差分的话 \(\text{GBST}\) 可能还要带一个两倍左右的常数。

当然你可以用一个类似树剖的写法,而不去差分成前缀链。那么这样的问题是在修改的两个点 \(x\)\(y\) 相遇时,相当于要做一个任意区间修改/查询,好在你只需做一次,那么可以接受单次 \(O(\log n)\) 的时间复杂度。还是在二叉搜索树上暴力跳,做一些分讨即可。

所以这样理论上树剖能做的问题,\(\text{GBST}\) 都是能做的,包括像链上 \(\min\) 这样的不可差分问题。不过它最经典的应用应该还是优化 \(\text{ddp}\),因为它本身就是前缀,又没有懒标记之类的东西,所以写起来非常方便。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/832482.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

考研打卡(15)

开局(15) 开始时间 2024-11-12 20:37:51 结束时间 2024-11-12 22:41:32现在在敷泥膜数据结构设已知一稀疏矩阵的三元组表为:(1,2,3),(1,6,1),(3,1,5),(3,2,-1) (5,4,5),(5,1,-3),则其转置矩阵的三元组表中的第三个三元组为____(山东大学2013年) A (2,…

25. 使用MySQL之使用触发器

1. 触发器 MySQL语句在需要时被执行,存储过程也是如此。但是,如果你想要某条语句(或某些语句)在事件发生时自动执行,怎么办呢?例如:每当增加一个顾客到某个数据库表时,都检查其电话号码格式是 否正确,州的缩写是否为大写;每当订购一个产品时,都从库存数量中减去订购…

Mysql表分区实操

创建分区表 create table user(id int(11) not null,name varchar(32) not null) partition by range(id) ( partition p0 values less than(10), partition p1 values less than(20), partition p2 values less than(30), partition p3 values less than maxvalue )分区表数据…

【Azure Bot Service】部署NodeJS ChatBot代码到App Service中无法自动启动

问题描述 把NodeJS的ChatBot代码部署到App Service环境中,通过VS Code直接部署,显示部署成功。但是通过URL访问时候,却是 :( Application Error 。问题解答 App Service遇见Application Error,第一步,查看日志。 发现启动时候遇见: 2024-11-12T12:22:40.366223350Z Error…

Chrome如何卸载掉插件?如此简单!

前言 我之前由于好奇,一下子装了很多的chrome插件,这些插件虽然都有很多功能,但是,由于我的电脑性能不佳,浏览器一下子装这么多插件,有点带不动了。所以,我就想到了卸载谷歌浏览器,哦不,是卸载浏览器里面的插件。今天,我就来介绍下,如何在chrome里面卸载一些不常用的…

rocky9如何重置root密码

rocky9如何重置root密码 第一步第二步 在箭头指的地方把 "ro" 改成 "rw"另一个箭头指的地方写上如下内容:init=/bin/bash如图所示第三步 Ctrl + x 按当前的配置启动第四步 输入命令root 用户密码直接输入以下内容passwd如果更改其他用户第五步 最为关键的…

多校A层冲刺NOIP2024模拟赛21

多校A层冲刺NOIP2024模拟赛21\(T1\) A. 送信卒 \(90pts/100pts\)部分分\(90pts\)设最后的可能的最短路中左右共移动了 \(d\) 次,上下共移动了 \(x\) 次。则等价于求 \(\min \{ x_{i}k+d_{i} \}=s\) 的解,观察到 \(d \in [0,\min(\left\lceil \frac{nm}{2} \right\rceil,s)]\)…

[论文阅读] ZePo: Zero-Shot Portrait Stylization with Faster Sampling

写在前面 原文:ZePo GitHub:Github ZePo 关键词:肖像风格化、扩散模型、零样本快速生成 阅读理由:对扩散模型的改进,可以实现零样本快速生成图像,学习一下思路以及实验设计 前置知识:LCM以及GithubLCM(找时间写一下),可参考LCM&CM,一致性蒸馏、图像质量评价速览…

2024.11.12 鲜花

P11270 【MX-S5-T4】魔法少女们 题解这世界那么多人 这世界有那么多人 人群里 敞着一扇门 我迷朦的眼睛里长存 初见你蓝色清晨 这世界有那么多人 多幸运 我有个我们 这悠长命运中的晨昏 常让我 望远方出神 灰树叶飘转在池塘 看飞机轰的一声去远乡 光阴的长廊 脚步声叫嚷 灯一亮…

Java代码实现行列转换

本代码想要达到的效果 测试完整代码如下(copy直接运行): public class TestConvert { public static void main(String[] args) { ArrayList<Attribute> sourceList = new ArrayList<>(); for (int i = 0; i < 3; i++) { for (i…

SS241112A. 定向越野(walk)

这道题目要求找到从起点出发遍历所有点并回到起点的最小路径长度,路径必须直角拐弯。文章证明了答案是有理数,并通过直观分析和大胆猜测得出初始方向必须是某一个向量的方向,最终使用状压 DP 解决问题,时间复杂度为 $O(n^4 2^n)$。SS241112A. 定向越野(walk) 题意 给你 \…

第六课 Python之模块

一、模块的介绍 (1)python模块,是一个python文件,以一个.py文件,包含了python对象定义和pyhton语句 (2)python对象定义和python语句 (3)模块让你能够有逻辑地组织你的python代码段。 (4)把相关的代码分配到一个模块里能让你的代码更好用,更易懂 (5)模块能定义函数…