树链剖分 学习笔记
树链剖分的思想及能解决的问题
树链剖分用于将树分割成若干条链的形式,以维护树上路径的信息。
具体来说,将整棵树剖分为若干条链,使它组合成线性结构,然后用其他的数据结构维护信息。
树链剖分(树剖 / 链剖)有多种形式,如 重链剖分,长链剖分 和用于 Link/cut Tree 的剖分(有时被称作「实链剖分
重链剖分可以将树上的任意一条路径划分成不超过
重链剖分还能保证划分出的每条链上的节点 DFS 序连续,因此可以方便地用一些维护序列的数据结构(如线段树)来维护树上路径的信息。
如:
- 修改 树上两点之间的路径上 所有点的值。
- 查询 树上两点之间的路径上 节点权值的 和 / 极值 / 其它(在序列上可以用数据结构维护,便于合并的信息)。
除了配合数据结构来维护树上路径信息,树剖还可以用来
重链剖分
我们给出一些定义:
定义 重子节点 表示其子节点中子树最大的子结点。如果有多个子树最大的子结点,取其一。如果没有子节点,就无重子节点。
定义 轻子节点 表示剩余的所有子结点。
从这个结点到重子节点的边为 重边。
到其他轻子节点的边为 轻边。
若干条首尾衔接的重边构成 重链。
把落单的结点也当作重链,那么整棵树就被剖分成若干条重链。
如图:
实现
树剖的实现分两个 DFS 的过程。伪代码如下:
第一个 DFS 记录每个结点的父节点(father
第二个 DFS 记录所在链的链顶(top,应初始化为结点本身
以下为代码实现。
我们先给出一些定义:
表示节点 在树上的父亲。 表示节点 在树上的深度。 表示节点 的子树的节点个数。 表示节点 的 重儿子。 表示节点 所在 重链 的顶部节点(深度最小 ) 。 表示节点 的 DFS 序,也是其在线段树中的编号。 表示 DFS 序所对应的节点编号,有 。
我们进行两遍 DFS 预处理出这些值,其中第一次 DFS 求出
void dfs1(int o) {
son[o] = -1;
siz[o] = 1;
for (int j = h[o]; j; j = nxt[j])
if (!dep[p[j]]) {
dep[p[j]] = dep[o] + 1;
fa[p[j]] = o;
dfs1(p[j]);
siz[o] += siz[p[j]];
if (son[o] == -1 || siz[p[j]] > siz[son[o]])
son[o] = p[j];
}
}
void dfs2(int o, int t) {
top[o] = t;
cnt++;
dfn[o] = cnt;
rnk[cnt] = o;
if (son[o] == -1)
return;
dfs2(son[o],
t); // 优先对重儿子进行 DFS,可以保证同一条重链上的点 DFS 序连续
for (int j = h[o]; j; j = nxt[j])
if (p[j] != son[o] && p[j] != fa[o])
dfs2(p[j], p[j]);
}
重链剖分的性质
树上每个节点都属于且仅属于一条重链。
重链开头的结点不一定是重子节点(因为重边是对于每一个结点都有定义的
所有的重链将整棵树 完全剖分。
在剖分时 重边优先遍历,最后树的 DFS 序上,重链内的 DFS 序是连续的。按 DFN 排序后的序列即为剖分后的链。
一颗子树内的 DFS 序是连续的。
可以发现,当我们向下经过一条 轻边 时,所在子树的大小至少会除以二。
因此,对于树上的任意一条路径,把它拆分成从 LCA 分别向两边往下走,分别最多走
常见应用
路径上维护
用树链剖分求树上两点路径权值和,伪代码如下:
链上的 DFS 序是连续的,可以使用线段树、树状数组维护。
每次选择深度较大的链往上跳,直到两点在同一条链上。
同样的跳链结构适用于维护、统计路径上的其他信息。
子树维护
有时会要求,维护子树上的信息,譬如将以
在 DFS 搜索的时候,子树中的结点的 DFS 序是连续的。
每一个结点记录 bottom 表示所在子树连续区间末端的结点。
这样就把子树信息转化为连续的一段区间信息。
求最近公共祖先
不断向上跳重链,当跳到同一条重链上时,深度较小的结点即为 LCA。
向上跳重链时需要先跳所在重链顶端深度较大的那个。
参考代码:
int lca(int u, int v) {
while (top[u] != top[v]) {
if (dep[top[u]] > dep[top[v]])
u = fa[top[u]];
else
v = fa[top[v]];
}
return dep[u] > dep[v] ? v : u;
}
怎么有理有据地卡树剖
一般情况下树剖的
常数不满很难卡,如果要卡只能建立二叉树深度低。 于是我们可以考虑折中方案。
我们建立一颗
个节点的二叉树。对于每个节点到其儿子的边,我们将其替换成一条长度为 的链。 这样子我们可以将随机询问轻重链切换次数卡到平均
次,同时有 的深度。 加上若干随机叶子看上去可以卡树剖。但是树剖常数小有可能卡不掉。
例题
「ZJOI2008」树的统计
题目大意
对一棵有
- 修改单个节点的权值;
- 查询
到 的路径上的最大权值; - 查询
到 的路径上的权值之和。
保证
解法
根据题面以及以上的性质,你的线段树需要维护三种操作:
- 单点修改;
- 区间查询最大值;
- 区间查询和。
单点修改很容易实现。
由于子树的 DFS 序连续(无论是否树剖都是如此
问题是如何修改 / 查询两个节点之间的路径。
考虑我们是如何用 倍增法求解 LCA 的。首先我们 将两个节点提到同一高度,然后将两个节点一起向上跳。对于树链剖分也可以使用这样的思想。
在向上跳的过程中,如果当前节点在重链上,向上跳到重链顶端,如果当前节点不在重链上,向上跳一个节点。如此直到两节点相同。沿途更新 / 查询区间信息。
对于每个询问,最多经过
给出一种代码实现:
// st 是线段树结构体
int querymax(int x, int y) {
int ret = -inf, fx = top[x], fy = top[y];
while (fx != fy) {
if (dep[fx] >= dep[fy])
ret = max(ret, st.query1(1, 1, n, dfn[fx], dfn[x])),
x = fa[fx];
else
ret = max(ret, st.query1(1, 1, n, dfn[fy], dfn[y])),
y = fa[fy];
fx = top[x];
fy = top[y];
}
if (dfn[x] < dfn[y])
ret = max(ret, st.query1(1, 1, n, dfn[x], dfn[y]));
else
ret = max(ret, st.query1(1, 1, n, dfn[y], dfn[x]));
return ret;
}
Nauuo and Binary Tree
这是一道交互题,也是树剖的非传统应用。
题目大意
有一棵以
节点数不超过
解法
首先可以通过
然后考虑按深度从小到大确定每个节点的父亲,这样的话确定一个节点的父亲时其所有祖先一定都是已知的。
确定一个节点的父亲之前,先对树已知的部分进行重链剖分。
假设我们需要在子树
其中红色虚线是一条重链,
这样的话,如果
时间复杂度
具体地,设
长链剖分
长链剖分本质上就是另外一种链剖分方式。
定义 重子节点 表示其子节点中子树深度最大的子结点。如果有多个子树最大的子结点,取其一。如果没有子节点,就无重子节点。
定义 轻子节点 表示剩余的子结点。
从这个结点到重子节点的边为 重边。
到其他轻子节点的边为 轻边。
若干条首尾衔接的重边构成 重链。
把落单的结点也当作重链,那么整棵树就被剖分成若干条重链。
如图(这种剖分方式既可以看成重链剖分也可以看成长链剖分
长链剖分实现方式和重链剖分类似,这里就不再展开。
常见应用
首先,我们发现长链剖分从一个节点到根的路径的轻边切换条数是
如何构造数据将轻重边切换次数卡满
我们可以构造这么一颗二叉树 T:
假设构造的二叉树参数为
。 若
, 则在左儿子构造一颗参数为 的二叉树,在右儿子构造一个长度为 的链。 若
, 则我们可以直接构造一个单独叶节点,并且结束调用。 这样子构造一定可以将单独叶节点到根的路径全部为轻边且需要
级别的节点数。 取
即可。
长链剖分优化 DP
一般情况下可以使用长链剖分来优化的 DP 会有一维状态为深度维。
我们可以考虑使用长链剖分优化树上 DP。
具体的,我们每个节点的状态直接继承其重儿子的节点状态,同时将轻儿子的 DP 状态暴力合并。
CF 1009F
我们设
表示在子树 i 内,和 i 距离为 j 的点数。 直接暴力转移时间复杂度为
我们考虑每次转移我们直接继承重儿子的 DP 数组和答案,并且考虑在此基础上进行更新。
首先我们需要将重儿子的 DP 数组前面插入一个元素 1, 这代表着当前节点。
然后我们将所有轻儿子的 DP 数组暴力和当前节点的 DP 数组合并。
注意到因为轻儿子的 DP 数组长度为轻儿子所在重链长度,而所有重链长度和为
。 也就是说,我们直接暴力合并轻儿子的总时间复杂度为
。 注意,一般情况下 DP 数组的内存分配为一条重链整体分配内存,链上不同的节点有不同的首位置指针。
DP 数组的长度我们可以根据子树最深节点算出。
注意,一般情况下 DP 数组的内存分配为一条重链整体分配内存,链上不同的节点有不同的首位置指针。
DP 数组的长度我们可以根据子树最深节点算出。
当然长链剖分优化 DP 技巧非常多,包括但是不仅限于打标记等等。这里不再展开。
长链剖分求 k 级祖先
即询问一个点向父亲跳
首先我们假设我们已经预处理了每一个节点的
现在我们假设我们找到了询问节点的
我们考虑求出其所在重链的节点并且按照深度列入表格。假设重链长度为
同时我们在预处理的时候找到每条重链的根节点的
根据长链剖分的性质,
预处理需要倍增出
预处理复杂度