树剖小结(简述)

比较恶心,特别是线段树
题目链接:P3384 【模板】重链剖分
详解见:大佬博客
我的代码:

\(Code\):

#include
#include
using namespace std;
const int MAXN=100005;
int mod,n,m,root;
int f[MAXN],dep[MAXN],son[MAXN],tot[MAXN];
int t[MAXN],b[MAXN],id[MAXN],c=0,top[MAXN];
struct edge
{
    int to,nxt;
}e[MAXN<<1];
int head[MAXN],cnt=0;
void swap(int *a,int *b){int t=*a;*a=*b,*b=t;}
void add(int u,int v)
{
    e[++cnt].to=v;
    e[cnt].nxt=head[u];
    head[u]=cnt;
}
int dfs1(int cur,int fa,int deep)
{
    dep[cur]=deep;
    f[cur]=fa;
    int maxn=-1;
    tot[cur]=1;
    for(int i=head[cur];i;i=e[i].nxt)
    {
        int j=e[i].to;
        if(!(j^fa)) continue;
        tot[cur]+=dfs1(j,cur,deep+1);
        if(tot[j]>maxn) maxn=tot[j],son[cur]=j;
    }
    return tot[cur];
}
void dfs2(int cur,int topf)
{
    id[cur]=++c;
    t[c]=b[cur];
    top[cur]=topf;
    if(!son[cur]) return;
    dfs2(son[cur],topf);
    for(int i=head[cur];i;i=e[i].nxt)
    {
        int j=e[i].to;
        if(!id[j]) dfs2(j,j);
    }
}
struct node
{
    int l,r,sum,lazy;
    node()
    {
        l=r=sum=lazy=0;
    }
}a[MAXN<<2];
void update(int k){a[k].sum=(a[k<<1].sum+a[k<<1|1].sum)%mod;}
void build(int k,int l,int r)
{
    int mid=(l+r)>>1;
    a[k].l=l,a[k].r=r;
    if(l==r)
    {
        a[k].sum=t[l];
        return;
    }
    build(k<<1,l,mid),build(k<<1|1,mid+1,r);
    update(k);
}
void lazydown(int k)
{
    if(a[k].l==a[k].r)
    {
        a[k].lazy=0;
        return;
    }
    a[k<<1].sum=(a[k<<1].sum+(a[k<<1].r-a[k<<1].l+1)*a[k].lazy%mod)%mod;
    a[k<<1|1].sum=(a[k<<1|1].sum+(a[k<<1|1].r-a[k<<1|1].l+1)*a[k].lazy%mod)%mod;
    a[k<<1].lazy=(a[k<<1].lazy+a[k].lazy)%mod;
    a[k<<1|1].lazy=(a[k<<1|1].lazy+a[k].lazy)%mod;
    a[k].lazy=0;
}
void change(int k,int l,int r,int x)
{
    int mid=(a[k].l+a[k].r)>>1;
    if(a[k].l==l&&a[k].r==r)
    {
        a[k].sum=(a[k].sum+(a[k].r-a[k].l+1)*x%mod)%mod;
        a[k].lazy=(a[k].lazy+x)%mod;
        return;
    }
    if(a[k].lazy) lazydown(k);
    if(r<=mid) change(k<<1,l,r,x);
    else if(l>mid) change(k<<1|1,l,r,x);
    else change(k<<1,l,mid,x),change(k<<1|1,mid+1,r,x);
    update(k);
}
int query(int k,int l,int r)
{
    int mid;
    if(a[k].lazy) lazydown(k);
    if(a[k].l==l&&a[k].r==r) return a[k].sum%mod;
    mid=(a[k].l+a[k].r)>>1;
    if(r<=mid) return query(k<<1,l,r)%mod;
    if(l>=mid+1) return query(k<<1|1,l,r)%mod;
    return (query(k<<1,l,mid)+query(k<<1|1,mid+1,r))%mod;
}
void tree_add(int l,int r,int x)
{
    while(top[l]!=top[r])
    {
        if(dep[top[l]]dep[r]) swap(l,r);
    change(1,id[l],id[r],x);
    return;
}

int tree_query(int l,int r)
{
    int ans=0;
    while(top[l]!=top[r])
    {
        if(dep[top[l]]dep[r]) swap(l,r);
    ans=(ans+query(1,id[l],id[r]))%mod;
    return ans%mod;
}
void son_add(int k,int x){change(1,id[k],id[k]+tot[k]-1,x);}
int son_query(int k){return query(1,id[k],id[k]+tot[k]-1);} 
int u,v,flag,x,y,z;
int main()
{
    scanf("%d%d%d%d",&n,&m,&root,&mod);
    for(int i=1;i<=n;i++) scanf("%d",&b[i]);
    for(int i=1;i

还是写炸了线段树,因为左右端点反了只有\(30pts\)(数据水?)
还看了\(bi\)站视频,
主要思想是根据\(dfs\)序使链和子树映射在线段树上,使得计算的相关点重新编号后在一个区间中,可证明时间复杂度为\(O(mlog^2n)\)似乎懂了
就这样

你可能感兴趣的:(树剖小结(简述))