树链剖分基本步骤

总体步骤: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

转载于:https://www.cnblogs.com/moyiii-/p/7182863.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值