BZOJ1984月下“毛景树” 树链剖分+线段树

Posted CQzhangyu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了BZOJ1984月下“毛景树” 树链剖分+线段树相关的知识,希望对你有一定的参考价值。

【BZOJ1984】月下“毛景树”

Description

毛毛虫经过及时的变形,最终逃过的一劫,离开了菜妈的菜园。 毛毛虫经过千山万水,历尽千辛万苦,最后来到了小小的绍兴一中的校园里。爬啊爬~爬啊爬~~毛毛虫爬到了一颗小小的“毛景树”下面,发现树上长着他最爱吃的毛毛果~~~ “毛景树”上有N个节点和N-1条树枝,但节点上是没有毛毛果的,毛毛果都是长在树枝上的。但是这棵“毛景树”有着神奇的魔力,他能改变树枝上毛毛果的个数: ? Change k w:将第k条树枝上毛毛果的个数改变为w个。 ? Cover u v w:将节点u与节点v之间的树枝上毛毛果的个数都改变为w个。 ? Add u v w:将节点u与节点v之间的树枝上毛毛果的个数都增加w个。 由于毛毛虫很贪,于是他会有如下询问: ? Max u v:询问节点u与节点v之间树枝上毛毛果个数最多有多少个。

Input

第一行一个正整数N。 接下来N-1行,每行三个正整数Ui,Vi和Wi,第i+1行描述第i条树枝。表示第i条树枝连接节点Ui和节点Vi,树枝上有Wi个毛毛果。 接下来是操作和询问,以“Stop”结束。

Output

对于毛毛虫的每个询问操作,输出一个答案。

Sample Input

4
1 2 8
1 3 7
3 4 9
Max 2 4
Cover 2 4 5
Add 1 4 10
Change 1 16
Max 2 4
Stop

Sample Output

9
16

【Data Range】
1<=N<=100,000,操作+询问数目不超过100,000。
保证在任意时刻,所有树枝上毛毛果的个数都不会超过10^9个。

题解:树剖+线段树裸题,不过注意是边权,把边权转换成点权,去掉最上面的点即可。还要注意一下cover和add两个标签的处理顺序。

#include <stdio.h>
#include <string.h>
#include <iostream>
#define lson x<<1
#define rson x<<1|1
using namespace std;
const int maxn=150010;
int n,cnt,tot;
int s[maxn<<2],p[maxn],tc[maxn<<2],ts[maxn<<2];
int to[maxn<<1],next[maxn<<1],val[maxn<<1],v[maxn],u[maxn],fa[maxn],head[maxn];
int deep[maxn],size[maxn],son[maxn],top[maxn];
char str[10];
int readin()
{
    int ret=0,sig=1;    char gc;
    while(gc<0||gc>9)    sig=(gc==-)?-1:1,gc=getchar();
    while(gc>=0&&gc<=9)    ret=ret*10+gc-0,gc=getchar();
    return ret*sig;
}
void add(int a,int b,int c)
{
    to[cnt]=b;
    val[cnt]=c;
    next[cnt]=head[a];
    head[a]=cnt++;
}
void dfs1(int x)
{
    size[x]=1;
    for(int i=head[x];i!=-1;i=next[i])
    {
        if(to[i]!=fa[x])
        {
            fa[to[i]]=x;
            deep[to[i]]=deep[x]+1;
            u[to[i]]=val[i];
            dfs1(to[i]);
            size[x]+=size[to[i]];
            if(size[to[i]]>size[son[x]])    son[x]=to[i];
        }
    }
}
void dfs2(int x,int tp)
{
    top[x]=tp;
    p[x]=++tot;
    v[p[x]]=u[x];
    if(son[x])    dfs2(son[x],tp);
    for(int i=head[x];i!=-1;i=next[i])
        if(to[i]!=son[x]&&to[i]!=fa[x])
            dfs2(to[i],to[i]);
}
void pushup(int x)
{
    s[x]=max(s[lson],s[rson]);
}
void pushdown(int x)
{
    if(tc[x]!=-1)    //注意顺序
    {
        s[lson]=s[rson]=tc[lson]=tc[rson]=tc[x];
        ts[lson]=ts[rson]=0;
        tc[x]=-1;
    }
    if(ts[x])
    {
        s[lson]+=ts[x],s[rson]+=ts[x],ts[lson]+=ts[x],ts[rson]+=ts[x];
        ts[x]=0;
    }
}
void build(int l,int r,int x)
{
    if(l==r)
    {
        s[x]=v[l];
        return ;
    }
    int mid=l+r>>1;
    build(l,mid,lson),build(mid+1,r,rson);
    pushup(x);
}
int query(int l,int r,int x,int a,int b)
{
    pushdown(x);
    if(a<=l&&r<=b)    return s[x];
    int mid=l+r>>1;
    if(b<=mid)    return query(l,mid,lson,a,b);
    if(a>mid)    return query(mid+1,r,rson,a,b);
    return max(query(l,mid,lson,a,b),query(mid+1,r,rson,a,b));
}
void upcover(int l,int r,int x,int a,int b,int c)
{
    pushdown(x);
    if(a<=l&&r<=b)
    {
        s[x]=tc[x]=c;
        return ;
    }
    int mid=l+r>>1;
    if(b<=mid)    upcover(l,mid,lson,a,b,c);
    else if(a>mid)    upcover(mid+1,r,rson,a,b,c);
    else    upcover(l,mid,lson,a,b,c),upcover(mid+1,r,rson,a,b,c);
    pushup(x);
}
void updata(int l,int r,int x,int y,int c)
{
    if(l==r)
    {
        s[x]=c;
        return ;
    }
    pushdown(x);
    int mid=l+r>>1;
    if(y<=mid)    updata(l,mid,lson,y,c);
    else    updata(mid+1,r,rson,y,c);
    pushup(x);
}
void upadd(int l,int r,int x,int a,int b,int c)
{
    pushdown(x);
    if(a<=l&&r<=b)
    {
        s[x]+=c,ts[x]+=c;
        return ;
    }
    int mid=l+r>>1;
    if(b<=mid)    upadd(l,mid,lson,a,b,c);
    else if(a>mid)    upadd(mid+1,r,rson,a,b,c);
    else upadd(l,mid,lson,a,b,c),upadd(mid+1,r,rson,a,b,c);
    pushup(x);
}
void getmax()
{
    int x=readin(),y=readin(),ans=0;
    while(top[x]!=top[y])
    {
        if(deep[top[x]]<deep[top[y]])    swap(x,y);
        ans=max(ans,query(1,n,1,p[top[x]],p[x]));
        x=fa[top[x]];
    }
    if(deep[x]>deep[y])    swap(x,y);
    if(x!=y)    ans=max(ans,query(1,n,1,p[x]+1,p[y]));    //p[x]要+1,取下面那一个点
    printf("%d\n",ans);
}
void Cover()
{
    int x=readin(),y=readin(),z=readin();
    while(top[x]!=top[y])
    {
        if(deep[top[x]]<deep[top[y]])    swap(x,y);
        upcover(1,n,1,p[top[x]],p[x],z);
        x=fa[top[x]];
    }
    if(deep[x]>deep[y])    swap(x,y);
    if(x!=y)    upcover(1,n,1,p[x]+1,p[y],z);
}
void Change()
{
    int x=readin(),y=readin();
    updata(1,n,1,max(p[to[x*2-2]],p[to[x*2-1]]),y);
}
void Add()
{
    int x=readin(),y=readin(),z=readin();
    while(top[x]!=top[y])
    {
        if(deep[top[x]]<deep[top[y]])    swap(x,y);
        upadd(1,n,1,p[top[x]],p[x],z);
        x=fa[top[x]];
    }
    if(deep[x]>deep[y])    swap(x,y);
    if(x!=y)    upadd(1,n,1,p[x]+1,p[y],z);
}
int main()
{
    n=readin();
    int i,a,b,c;
    memset(head,-1,sizeof(head));
    memset(tc,-1,sizeof(tc));
    for(i=1;i<n;i++)
    {
        a=readin(),b=readin(),c=readin();
        add(a,b,c),add(b,a,c);
    }
    deep[1]=1;
    dfs1(1);
    dfs2(1,1);
    build(1,n,1);
    while(scanf("%s",str))
    {
        switch(str[1])
        {
            case a:getmax();    break;
            case o:Cover();    break;
            case d:Add();    break;
            case h:Change();    break;
            case t:return 0;
        }
    }
}

 

以上是关于BZOJ1984月下“毛景树” 树链剖分+线段树的主要内容,如果未能解决你的问题,请参考以下文章

BZOJ1984 月下“毛景树”

题解 P4315 月下“毛景树”

bzoj1984 月下“毛景树”

[BZOJ1984] 月下“毛景树”

P4315 月下“毛景树”

[luogu4315]月下“毛景树”