博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
模板:树链剖分
阅读量:6227 次
发布时间:2019-06-21

本文共 1279 字,大约阅读时间需要 4 分钟。

“如果你会了树上dp,还会线段树……”

“没错!我都会啊!”

“……那你为什么写不出树链剖分?”

“???”

——by勇者和路由器的对话,今天二位仍然过得十分愉快

————————————————————————————

因为路由器编不出来什么好题面了,所以就扔上来了一个模板题然后和勇者去玩了。

(下面有树链剖分的板子代码+注释,这里只提供讲解)

我们发现题中要求的内容类似于线段树:单点修改,区间询问。

但是,这是一棵树啊!我们怎么才能在树上建一个只适用于一维的数据结构呢?

我们要抛弃线段树吗?

……

那么我们要试图把树拍扁成一维的吗?

……只能这样了。

其实拍扁成一维并不难想,考虑当树为一条链的时候吗,我们就直接上线段树即可。

那么类比一棵完整的树时,我们就把它分解成一条一条链然后拼在一起线段树维护即可。

关键问题在于要如何分解成链才能使得我们查询既快捷又方便呢?

这里当然就是树链剖分的活啦!

 

树链剖分:

概念:

定义size(X)为以X为根的子树的节点个数。 令V为U的儿子节点中size值最大的节点,那么边(U,V)被称为重边,树中重边之外的边被称为轻边。

我们称某条路径为重路径,当且仅当它全部由重边组成。

性质:

性质 1:轻边(U,V),size(V)<=size(U)/2。

性质 2:从根到某一点的路径上,不超过O(logN)条轻边,不超过O(logN)条重路径。

对于性质1,我们肉眼观察法和反证法都能解决。

对于性质2就不是那么明显了,我们来证明一下:

由性质1可知,每经过一条轻边,子树的节点个数至少减少一半,所以至多经过 O (log n ) 条轻边。

而进入(或从……出去)一条重路径,一定需要经过一条轻边,所以至多经过 O (log n ) 条重路径。

有了以上两个性质之后,我们就可以发现这种分法的优越性了,我们仅仅只需要搜大概logn级别即可。

细节:

预处理:

我们具体需要求出7个值,分别为:

对于节点u:

父亲fa;

深度dep;

子树节点数size(又叫重量);

重儿子son;

所在重路径的顶部节点top;

在序列的位置pos(下标)。

对于序列的一个下标:

对应树的位置idx。

——————————————————

前四个朴素dfs即可解决,后三个根据节点的重儿子再dfs即可解决。

注意:我们的目的是为了将树分解成重路径,所以第二次dfs建序列的时候要先加重儿子再管其他节点。

 

求值:

我们将u到v的路径分解成:

当u与v不在同一个重路径时:

(u所在的部分重路径)+(top[u]到top[v])+(v所在的部分重路径)。

当u与v在同一个重路径时(显然不需要分解)

按照上面的方法递归并且不断求出这些段的值完后汇总即可。

那么我们就想先跳u为top[u]还是跳v为top[v]——方法就是,为了防止跳大了,跳得越少越好(比较top[u]和top[v]的dep即可)。

 

例题(难度递增):

转载于:https://www.cnblogs.com/luyouqi233/p/7886709.html

你可能感兴趣的文章
Yii用ajax实现无刷新检索更新CListView数据
查看>>
JDBC的事务
查看>>
Io流的概述
查看>>
App 卸载记录
查看>>
JavaScript变量和作用域
查看>>
开源SIP服务器加密软件NethidPro升级
查看>>
百度页面分享插件源代码
查看>>
《别做正常的傻瓜》的一些读书心得
查看>>
作业:实现简单的shell sed替换功能和修改haproxy配置文件
查看>>
spring配置多数据源问题
查看>>
Altium 拼板方法以及 注意的 地方
查看>>
简明Linux命令行笔记:tail
查看>>
PMP考试的过与只是
查看>>
java 监控 收集资料3(收集中)
查看>>
Apache Pulsar中的地域复制,第1篇:概念和功能
查看>>
getRealPath()和getContextPath()的区别
查看>>
Hadoop MapReduce编程 API入门系列之wordcount版本2(六)
查看>>
一个页面标题和过滤输出的解决方案(上)
查看>>
python pip install 出现 OSError: [Errno 1] Operation not permitted
查看>>
oracle12C 重做日志
查看>>