点分治
点分治和树剖有一个相同点,就是算法都是从暴力经过一小步优化而来,但就是这一小步,让算法复杂度有了质的飞跃。 下面来讲一下这个算法的原理。 概览 其实“点分治”也是一个名不副实的算法,看完下面的讲述你就会发现这个算法根“分治”几乎无关。 点分治是专门解决树上路径统计的一种算法(树剖则是解决对树上路径的判断的算法)。 点分治的思想很朴素,就是DFS一下 $\text{lca}$ 的节点编号 $x$,然后统计 $\text{lca}$ 值刚好为 $x$ 的、满足题目条件的路径的数量,并累加到答案。 但直接进行统计还是炸裂的,点分治则是对这种统计方式做了一步优化 解决题型 上面说了,「点分治是专门解决树上路径统计的一种算法」,但并不是所有的树上路径统计都可以用点分治解决。 其实看完下面的实现,你就能知道它解决什么类问题,不过这里还是说一下。 就是首先,这个条件是形如“总和比某个值低”、“总和是某个值的倍数”等“可累加”的条件。 其中“可累加”指的是,如果知道了 $A$ 部分的某个值,和 $B$ 部分的某个值,那么 $A$ 和 $B$ 部分合并后的部分是否合法也一定知道。 但其他的条件目前还没有太明确,得自己去总结。 算法实现 下面说一下点分治的实现。 大体思路 我们拿“路径长度(路径上的边的边权之和)刚好为 $k$”问题举例。 注:原题链接P3806 就是我们考虑,当DFS到某个点 $x$ 的时候,就如上所说,设 $x$ 为 $\text{lca}$ 值。 然后,我们按顺序遍历所有子节点 $to$。 由于只要两个点 $y$、$z$ 满足 $y$ 和 $z$ 分别在 $x$ 的两个不同儿子及其子树内,那么 $\text{lca}(y,z)=x$。 并且所有满足条件 $\text{lca}(y,z)=x$ 的对 $(y,z)$ 只有满足上述条件的对。 所以,我们可以通过(另一种)DFS(方式)求出 $to$ 到 ( 所有 $to$ 及其子树内的点 ) 之间的距离(加上 $x \to to$ 的边权)的集合 $T$。 ...