码迷,mamicode.com
首页 > 其他好文 > 详细

树链剖分模板

时间:2016-05-07 11:20:16      阅读:225      评论:0      收藏:0      [点我收藏+]

标签:

树链剖分的详解这里写得很好。我的标程与它的做法一样。
树链剖分的作用相当于在树上做线段树。
模板题
【ZJOI2008】树的统计
在一颗树上,区间查询,单点修改。
I. CHANGE u t : 把结点u的权值改为t
II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值
III. QSUM u v: 询问从点u到点v的路径上的节点的权值和

标程

#include <cstdio>
#include <cmath>
#include <algorithm>
#include <cstdlib>
#include <cstring>
#include <iostream>
#define fo(i,a,b) for(ll i=a;i<=b;i++)
#define max(a,b) ((a)>(b)?(a):(b))
#define ll long long
#define N 30100
int n,a[N],last[N*10],next[N*10],to[N*10],tot,size[N],deep[N],dfn[N],son[N],fa[N],top[N],an;
struct note
{
    int mx,sum;
};
note tree[N*10];
void putin(int x,int y)
{
    next[++tot]=last[x];last[x]=tot;to[tot]=y;
}
void dg1(int x)
{
    size[x]=1;int jy=0;
    for(int i=last[x];i;i=next[i])
    {
        int k=to[i];
        if (deep[k]!=0) continue;
        fa[k]=x;deep[k]=deep[x]+1;dg1(k);size[x]+=size[k];
        if (size[k]>jy) son[x]=k,jy=size[k];
    }
}
void dg2(int x,int y)
{
    dfn[x]=++tot;if (son[x]) top[son[x]]=top[x],dg2(son[x],x);
    for(int i=last[x];i;i=next[i])
    {
        int k=to[i];
        if (k==son[x] || k==y) continue;
        top[k]=k;dg2(k,x);
    }
}
void change(int v,int i,int j,int x,int y)
{
    if (i==j) { tree[v].mx=y;tree[v].sum=y;return;}
    int mid;mid=(i+j)/2;
    if (x<=mid) change(v*2,i,mid,x,y); else change(v*2+1,mid+1,j,x,y);
    tree[v].mx=max(tree[v*2].mx,tree[v*2+1].mx);
    tree[v].sum=tree[v*2].sum+tree[v*2+1].sum;
}
void get(int v,int i,int j,int x,int y,int jy)
{
    if (i==x && j==y){ if (jy==1) an+=tree[v].sum;else an=max(an,tree[v].mx);return;}
    int mid;mid=(i+j)/2;
    if (y<=mid) get(v*2,i,mid,x,y,jy);
    else if(x>mid) get(v*2+1,mid+1,j,x,y,jy);
         else get(v*2,i,mid,x,mid,jy),get(v*2+1,mid+1,j,mid+1,y,jy);
}
int lct(int x,int y,int jy)
{
    int ans,bz=1;
    ans=jy==1?0:-2147483647;
    while (x!=y)
    {
        int f1=top[x],f2=top[y];an=jy==1?0:-2147483647;
        if (f1!=f2)
        {
            if (deep[f1]>=deep[f2])
            {
                get(1,1,tot,dfn[f1],dfn[x],jy);
                x=fa[f1];
            }
            else
            {
                get(1,1,tot,dfn[f2],dfn[y],jy);
                y=fa[f2];
            }
        }
        else
        {
            if (deep[x]<deep[y]) get(1,1,tot,dfn[x],dfn[y],jy); else get(1,1,tot,dfn[y],dfn[x],jy);
            if (jy==1) ans+=an;else ans=max(ans,an);
            bz=0;
            break;
        }
        if (jy==1) ans+=an;else ans=max(ans,an);
    }
    if (x!=0 && x==y && bz) 
    {
        an=jy==1?0:-2147483647;
        get(1,1,tot,dfn[x],dfn[y],jy);
        if (jy==1) ans+=an;else ans=max(ans,an);
    }
    return ans;
}
int main()
{
    scanf("%d",&n);
    fo(i,1,n-1)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        putin(x,y);putin(y,x);
    }
    fo(i,1,n) scanf("%d",&a[i]);
    deep[1]=1;top[1]=1;dg1(1);
    tot=0;dg2(1,0);
    fo(i,1,n) change(1,1,tot,dfn[i],a[i]);
    int ac;scanf("%d\n",&ac);
    for(;ac;ac--)
    {
        char ch;scanf("%c",&ch);scanf("%c",&ch);
        int x,y;
        if (ch==‘M‘)
        {
            scanf("AX %d %d\n",&x,&y);
            printf("%d\n",lct(x,y,2));
        }             
        if (ch==‘S‘)
        {
            scanf("UM %d %d\n",&x,&y);
            printf("%d\n",lct(x,y,1));
        }
        if (ch==‘H‘)
        {
            scanf("ANGE %d %d\n",&x,&y);
            change(1,1,tot,dfn[x],y);
        }
    }
}

树链剖分模板

标签:

原文地址:http://blog.csdn.net/u011056504/article/details/51333164

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!