zoukankan      html  css  js  c++  java
  • [转] 虚树的概念

    转自xyz:

    给出一棵树.

    每次询问选择一些点,求一些东西.这些东西的特点是,许多未选择的点可以通过某种方式剔除而不影响最终结果.

    于是就有了建虚树这个技巧.....

    我们可以用log级别的时间求出点对间的lca....

    那么,对于每个询问我们根据原树的信息重新建树,这棵树中要尽量少地包含未选择节点. 这棵树就叫做虚树.

    接下来所说的"树"均指虚树,原来那棵树叫做"原树".

    构建过程如下:

    按照原树的dfs序号(记为dfn)递增顺序遍历选择的节点. 每次遍历节点都把这个节点插到树上.

    首先虚树一定要有一个根. 随便扯一个不会成为询问点的点作根.

    维护一个栈,它表示在我们已经(用之前的那些点)构建完毕的虚树上,以最后一个插入的点为端点的DFS链.

    设最后插入的点为p(就是栈顶的点),当前遍历到的点为x.我们想把x插入到我们已经构建的树上去.

    求出lca(p,x),记为lca.有两种情况:

      1.p和x分立在lca的两棵子树下.

      2.lca是p.

      (为什么lca不能是x?

       因为如果lca是x,说明dfn(lca)=dfn(x)<dfn(a),而我们是按照dfs序号遍历的,于是dfn(a)<dfn(x),矛盾.)

     对于第二种情况,直接在栈中插入节点x即可,不要连接任何边(后面会说为什么).

    对于第一种情况,要仔细分析.

    我们是按照dfs序号遍历的(因为很重要所以多说几遍......),有dfn(x)>dfn(p)>dfn(lca).

    这说明什么呢? 说明一件很重要的事:我们已经把lca所引领的子树中,p所在的子树全部遍历完了!

      简略的证明:如果没有遍历完,那么肯定有一个未加入的点h,满足dfn(h)<dfn(x),

            我们按照dfs序号递增顺序遍历的话,应该把h加进来了才能考虑x.

    这样,我们就直接构建lca引领的,p所在的那个子树. 我们在退栈的时候构建子树.

    p所在的子树如果还有其它部分,它一定在之前就构建好了(所有退栈的点都已经被正确地连入树中了),就剩那条链.

    如何正确地把p到lca那部分连进去呢?

    设栈顶的节点为p,栈顶第二个节点为q.

    重复以下操作:

      如果dfn(q)>dfn(lca),可以直接连边q->p,然后退一次栈.

      如果dfn(q)=dfn(lca),说明q=lca,直接连边lca->p,此时子树已经构建完毕.

      如果dfn(q)<dfn(lca),说明lca被p与q夹在中间,此时连边lca->q,退一次栈,再把lca压入栈.此时子树构建完毕.

        如果不理解这样操作的缘由可以画画图.....

    最后,为了维护dfs链,要把x压入栈. 整个过程就是这样.....

  • 相关阅读:
    软件需求与建模 复习笔记
    Autoware 笔记No.9,SSD车辆、行人(障碍物)识别(ssd vision detect)
    Autoware 笔记No.8 ENet 障碍物识别(vision segment ENet detect)
    Autoware 1.14 完整安装
    Autoware 笔记No.7, CNN障碍物检测(CNN LiDAR Baidu Object Segmenter)
    iOS 使用局部block处理接口依次调用需求
    阿里一面凉经
    Codeforces round #717 D.Cut(m询问求区间[L,R]能被至少分成多少个区间让每个小区间各数的乘积==各数的LCM)
    2018-2020 国家集训队论文选读
    GDOI 2021 游记
  • 原文地址:https://www.cnblogs.com/lidaxin/p/5113381.html
Copyright © 2011-2022 走看看