zoukankan      html  css  js  c++  java
  • 树链剖分基本步骤

    总体步骤:init();
              dfs1(1);
              dfs2(1,1);
              build(1,1,n);
              cl();
    
    dfs1 处理出size[sj](子节点数),son[sj](重儿子),dep[sj](深度),vi[sj](点权,把边权放在子节点上)。
    
    dfs2 处理出top[sj](所在链的链首,轻儿子从子节点重新开始,重儿子从父节点继承),id[sj](点出现的位置,用cnt标记),pos[sj](cnt位置的边连着哪一个点)。
          id[x]=++cnt;
          pos[cnt]=x;
          if(son[x])  dfs2(son[x],y);
          for(int i=h[x];i!=-1;i=b[i].ne)
            if(b[i].v!=fa[x]&&b[i].v!=son[x])  dfs2(b[i].v,b[i].v);
    
    build if(z==y) mx[x]=vi[pos[y]];
          这个位置链的最大值等于这个位置的链所连点的权值。一般线段树方法建树。和线段树有关的数组要开四倍四倍四倍。。。
    
    cl 分change和query
       Change 改变的边编号要按输入数据检索,从线段树上找到这条边上作为子节点的点改变它的点权。
                int now=(b1<<1)-1;
                int fr=b[now].u,to=b[now].v;
                if(fa[fr]==to)   change(id[fr],b2,1,1,n);
                else change(id[to],b2,1,1,n);
       Query  query函数即为一般的线段树查询,把不在一条链上的两个点向一条链上靠拢,时刻注意要求哪点在下层。如果本来就是点权,则id[x]不应该+1且id[x]==id[y]的情况是可行的。
           int fx=top[x],fy=top[y],res=1<<31;
           while(fx^fy)
           {
                if(dep[fx]<dep[fy])
                {
                    jh(x,y);
                    jh(fx,fy);
                }
                res=qd(res,query(id[fx],id[x],1,1,n));
                x=fa[fx];
                fx=top[x];
           }
           if(dep[x]>dep[y])  jh(x,y);
           if(id[x]<id[y])
               res=qd(res,query(id[x]+1,id[y],1,1,n));
           return res;
    
    零散的小函数 qd取较大值,jh取地址交换,add邻接表加边(要有u、v、w、ne)

    因为我太弱了不写就会很快忘掉QAQ
    南风知我意,吹梦到西洲。
  • 相关阅读:
    Redis源码剖析(十一)AOF持久化
    Redis源码剖析(一)内存管理zmalloc
    Redis源码剖析(七)压缩列表
    PHP基础课程学习总结
    php标记,变量,常量
    内容控件
    WPF 面板
    OpenSSL SSL_read: Connection was reset, errno 10054
    Java Web 项目学习(三) 发布帖子 AJAX
    控件类(ControlAndFont)
  • 原文地址:https://www.cnblogs.com/moyiii-/p/7182863.html
Copyright © 2011-2022 走看看