树链剖分

将一棵树的每个节点到它所有子节点中子树和(所包含的点的个数)最大的那个子节点的这条边标记为“重边”。

将其他的边标记为“轻边”。

若果一个非根节点的子树的大小不小于任意一个他兄弟节点的子数大小(若有多个就看心情选取其中的一个),那么它到它父节点的连边为重边,这个节点为重子节点,否则,它到它父节点的连边为轻边。

将一条全部由重边组成的链叫做重链。

(图中加粗的边为重边,未加粗的边为轻边,图取自www.baidu.com)

这样做有什么用呢?

如上图剖分后的树有这样的性质:

1.每个点都在一条重链中(轻子节点所在重链链顶是它本身)

2.每一条重链一定是自上而下(即不会再一条重链上出现两个深度相同的点)

3.任意一个节点到根节点的路径上最多有log2(n)条轻边和log2(n)条重链。

这样之后,我们可以按照优先级为“根节点>重子节点>轻子节点”的顺序进行两次dfs,O(n)预处理出dfs序,深度,每个节点所在重链的顶端,这样就能保证每一条重链中所有的点的dfs序中的位置都是自上而下连续而递增的。

然后,我们再用一个线段树O(log2(n))维护每一条重链和轻边上点的权值的单点修改,区间修改,区间和、区间最值之类的问题。

如果要处理路径上的问题,我们可以不断将两个点中所在重链链顶深度小的那个点直接跳到它所在重链链顶的父节点,并对这条重链进行操作(如果这个点是轻子节点那么就直接对这个点进行操作),直到这两个点到达了同一条重链,这是再将两个点之间的部分进行操作。

这样,我们每次完成对一条路径的操作或询问复杂度为log2(n)乘以 log2(m)(m为每条链的平均长度,实际上这个数通常较小)。

这里附上洛谷模板题和AC代码。

题目描述

如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作:

操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z

操作2: 格式: 2 x y 表示求树从x到y结点最短路径上所有节点的值之和

操作3: 格式: 3 x z 表示将以x为根节点的子树内所有节点值都加上z

操作4: 格式: 4 x 表示求以x为根节点的子树内所有节点值之和

输入输出格式

输入格式:

第一行包含4个正整数N、M、R、P,分别表示树的结点个数、操作个数、根节点序号和取模数(即所有的输出结果均对此取模)。

接下来一行包含N个非负整数,分别依次表示各个节点上初始的数值。

接下来N-1行每行包含两个整数x、y,表示点x和点y之间连有一条边(保证无环且连通)

接下来M行每行包含若干个正整数,每行表示一个操作,格式如下:

操作1: 1 x y z

操作2: 2 x y

操作3: 3 x z

操作4: 4 x

输出格式:

输出包含若干行,分别依次表示每个操作2或操作4所得的结果(对P取模)

输入输出样例

输入样例:

5 5 2 24
7 3 7 8 0
1 2
1 5
3 1
4 1
3 4 2
3 2 2
4 5
1 5 1 3
2 1 3
输出样例:

2
21

 #include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define mid (l+r>>1)
#define len (r-l+1)
#define M 100100
using namespace std;
LL read(){
LL nm=,oe=;char cw=getchar();
while(!isdigit(cw)) oe=cw=='-'?-oe:oe,cw=getchar();
while(isdigit(cw)) nm=nm*+(cw-''),cw=getchar();
return nm*oe;
}
LL n,m,f[M],fa[M],nt[M<<],to[M<<],tp[M],d[M],w[M];
LL cnt,cur,mod,rt,a,b,sz[M],gt[M],ed[M],s[M],num,c[M];
LL t[M<<],mk[M<<],add,typ;
bool fg[M];
void link(){nt[++cur]=f[a],f[a]=cur,to[cur]=b;}
void dfs1(LL x){
sz[x]=;
int sn=;
for(int i=f[x];i!=-;i=nt[i]){
if(to[i]==fa[x]) continue;
fa[to[i]]=x,d[to[i]]=d[x]+;
dfs1(to[i]),sz[x]+=sz[to[i]];
if(sn==) sn=i;
else if(sz[to[sn]]<sz[to[i]]) sn=i;
}
if(sn!=) swap(to[f[x]],to[sn]),fg[to[f[x]]]=true;
return;
}
void dfs2(int x){
if(fg[x]) tp[x]=tp[fa[x]];
else tp[x]=x;
gt[x]=++cnt,s[gt[x]]=x;
for(int i=f[x];i!=-;i=nt[i]){
if(to[i]==fa[x]) continue;
dfs2(to[i]);
}
ed[x]=cnt;
}
int build(int x,int l,int r){
if(l==r) return t[x]=w[s[l]];
return t[x]=(build(x<<,l,mid)+build(x<<|,mid+,r))%mod;
}
void pushdown(int x,int l,int r){
mk[x<<]+=mk[x],t[x<<]+=(mid-l+)*mk[x];
mk[x<<|]+=mk[x],t[x<<|]+=(r-mid)*mk[x];
mk[x]=;
}
void update(int x,int l,int r,int L,int R){
if(r<L||l>R) return;
if(L<=l&&r<=R){
mk[x]+=add;
t[x]+=len*add;
t[x]%=mod;
return;
}
pushdown(x,l,r);
update(x<<,l,mid,L,R);
update(x<<|,mid+,r,L,R);
t[x]=(t[x<<]+t[x<<|])%mod;
}
LL calc(int x,int l,int r,int L,int R){
if(r<L||R<l) return ;
if(L<=l&&r<=R) return t[x];
pushdown(x,l,r);
LL tot=calc(x<<,l,mid,L,R)+calc(x<<|,mid+,r,L,R);
t[x]=(t[x<<]+t[x<<|])%mod;
return tot%mod;
}
void change(){
int x=a,y=b;
while(tp[x]!=tp[y]){
if(d[tp[x]]<d[tp[y]]) swap(x,y);
update(,,n,gt[tp[x]],gt[x]),x=fa[tp[x]];
}
if(d[x]>d[y]) swap(x,y);
update(,,n,gt[x],gt[y]);
return;
}
LL ans(){
LL tot=0ll,x=a,y=b;
while(tp[x]!=tp[y]){
if(d[tp[x]]<d[tp[y]]) swap(x,y);
tot+=calc(,,n,gt[tp[x]],gt[x]),x=fa[tp[x]];
tot%=mod;
}
if(d[x]>d[y]) swap(x,y);
tot+=calc(,,n,gt[x],gt[y]);
return tot;
}
int main(){
n=read(),m=read(),rt=read(),mod=read();
for(int i=;i<=n;i++) w[i]=read(),f[i]=-,fg[i]=false;
for(int i=;i<n;i++){
a=read(),b=read();
link(),swap(a,b),link();
}
tp[rt]=fa[rt]=rt,d[rt]=;
dfs1(rt),dfs2(rt),build(,,n);
while(m--){
typ=read(),a=read();
if(typ==) b=read(),add=read(),change();
else if(typ==) b=read(),printf("%lld\n",ans()%mod);
else if(typ==) add=read(),update(,,n,gt[a],ed[a]);
else printf("%lld\n",calc(,,n,gt[a],ed[a])%mod);
}
return ;
}

本人代码风格较为奇怪,请大家见谅。

洛谷 P3384 【模板】树链剖分的更多相关文章

  1. [洛谷P3384] [模板] 树链剖分

    题目传送门 显然是一道模板题. 然而索引出现了错误,狂wa不止. 感谢神犇Dr_J指正.%%%orz. 建线段树的时候,第44行. 把sum[p]=bv[pos[l]]%mod;打成了sum[p]=b ...

  2. [luogu P3384] [模板]树链剖分

    [luogu P3384] [模板]树链剖分 题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点 ...

  3. 洛谷P3979 遥远的国度 树链剖分+分类讨论

    题意:给出一棵树,这棵树每个点有权值,然后有3种操作.操作一:修改树根为rt,操作二:修改u到v路径上点权值为w,操作三:询问以rt为根x子树的最小权值. 解法:如果没有修改树根操作那么这题就是树链剖 ...

  4. 洛谷 P4114 Qtree1 树链剖分

    目录 题面 题目链接 题目描述 输入输出格式 输入格式: 输出格式: 输入输出样例 输入样例: 输出样例: 说明 说明 思路 Change Query AC代码 总结 题面 题目链接 P4114 Qt ...

  5. 洛谷.4114.Qtree1(树链剖分)

    题目链接 模板题都错了这么多次.. //边权赋到点上 树剖模板 //注意LCA.链的顶端不能统计到答案! #include <cstdio> #include <cctype> ...

  6. 洛谷3384&bzoj1036树链剖分

    值得注意的是: 一个点的子树是存在一起的...也就是说我们修改子树的时候只用... /********************************************************* ...

  7. P3384 [模板] 树链剖分

    #include <bits/stdc++.h> using namespace std; typedef long long ll; int n, m, rt, mod, cnt, to ...

  8. luoguP3384 [模板]树链剖分

    luogu P3384 [模板]树链剖分 题目 #include<iostream> #include<cstdlib> #include<cstdio> #inc ...

  9. 【Luogu P3384】树链剖分模板

    树链剖分的基本思想是把一棵树剖分成若干条链,再利用线段树等数据结构维护相关数据,可以非常暴力优雅地解决很多问题. 树链剖分中的几个基本概念: 重儿子:对于当前节点的所有儿子中,子树大小最大的一个儿子就 ...

  10. 模板 树链剖分BFS版本

    //点和线段树都从1开始 //边使用vector vector<int> G[maxn]; ],num[maxn],iii[maxn],b[maxn],a[maxn],top[maxn], ...

随机推荐

  1. 手把手教你用npm发布一个包,详细教程

    我们已经实现了路由的自动化构建,但是我们可以看到,一大串代码怼在里面.当然你也可以说,把它封装在一个JS文件里面,然后使用require('./autoRoute.js')给引入进来,那也行.但是,为 ...

  2. CentOS7 Redis安装

    Redis介绍 1.安装Redis 官方下载地址:http://download.redis.io 使用Linux下载:wget http://download.redis.io/redis-stab ...

  3. Python二维数据分析

    一.numpy二维数组 1.声明 import numpy as np #每一个[]代表一行 ridership = np.array([ [ 0, 0, 2, 5, 0], [1478, 3877, ...

  4. asp.net调用Lodop实现页面打印或局部打印,可进行打印设置或预览

    <%@ Page Language="C#" AutoEventWireup="true" CodeFile="WebPrint.aspx.cs ...

  5. Javascript 面向对象编程—封装

      前  言 Javascript是一种基于对象(object-based)的语言,你遇到的所有东西几乎都是对象.但是,它又不是一种真正的面向对象编程(OOP)语言,因为它的语法中没有class(类) ...

  6. 干了这杯Java之LinkedList

    LinkedList和ArrayList一样实现了List接口 ArrayList内部为数组 LinkedList内外为双向链表 实现了Deque接口,双端列队的实现 图片来自Wiki 内部实现为No ...

  7. IOS应用FFMPEG库

    1.引用资源 build-ffmpeg  ffmpeg库生成 -sh开源地址: https://gist.github.com/m1entus/6983547 iFrameExtractor ffmp ...

  8. win10 uwp clone

    clone 可以用MemberwiseClone来复制一个类 但这个复制是浅复制,创建一个新的object然后复制值字段,对于引用就直接复制引用,不复制引用的本身,指向同样引用 如果要复制引用,可以使 ...

  9. win10 uwp 右击浮出窗在点击位置

    本文主要让MenuFlyout出现在我们右击位置. 我们一般使用的MenuFlyout写在前台,写在Button里面,但是可能我们的MenuFlyout显示的位置和我们想要的不一样. 通过使用后台写S ...

  10. 整理一批 国内外优秀设计团队 & 设计相关网站

    设计做不好,因为看得少!这里精心整理了一批国内外优秀设计团队的官网,以及同设计相关的网站.每个网站,我都浏览了一下,确保里面真的是有干货,并且保持一定的频率正常更新. [腾讯] 腾讯社交用户体验设计 ...