HYSBZ - 1036 树的统计Count (树链剖分+线段树单点更新,区间查询)

树的统计Count

一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w。我们将以下面的形式来要求你对这棵树完成
一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 I
II. QSUM u v: 询问从点u到点v的路径上的节点的权值和 注意:从点u到点v的路径上的节点包括u和v本身

Input

  输入的第一行为一个整数n,表示节点的个数。接下来n – 1行,每行2个整数a和b,表示节点a和节点b之间有
一条边相连。接下来n行,每行一个整数,第i行的整数wi表示节点i的权值。接下来1行,为一个整数q,表示操作
的总数。接下来q行,每行一个操作,以“CHANGE u t”或者“QMAX u v”或者“QSUM u v”的形式给出。 
对于100%的数据,保证1<=n<=30000,0<=q<=200000;中途操作中保证每个节点的权值w在-30000到30000之间。

Output

  对于每个“QMAX”或者“QSUM”的操作,每行输出一个整数表示要求输出的结果。

Sample Input

4 1 2 2 3 4 1 4 2 1 3 12 QMAX 3 4 QMAX 3 3 QMAX 3 2 QMAX 2 3 QSUM 3 4 QSUM 2 1 CHANGE 1 5 QMAX 3 4 CHANGE 3 6 QMAX 3 4 QMAX 2 4 QSUM 3 4

Sample Output

4 1 2 2 10 6 5 6 5 16

题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1036

思路:利用树链剖分将树分成若干条节点id编号连续的链,用线段树实现单点的更新,维护区间的最大值与和

注意:节点的权值可为负 所以要设为-inf 

代码:

#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<iostream>
using namespace std;
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
#define inf 0x3f3f3f3f
const int N=30005;
int first[N],tot,cnt,n,w[N],wt[N],sum[N<<2],maxx[N<<2];
int d[N],fa[N],siz[N],son[N],id[N],top[N],rk[N];
struct node
{
    int v,nex;
}e[N<<1];
void adde(int u,int v)
{
    e[tot].v=v;
    e[tot].nex=first[u];
    first[u]=tot++;
}
void init()
{
    tot=cnt=0;
    memset(first,-1,sizeof(first));
    memset(son,0,sizeof(son));
}

void dfs1(int u,int pre,int depth) //处理 d,fa,siz,son 数组
{
    d[u]=depth;
    fa[u]=pre;
    siz[u]=1;
    for(int i=first[u];~i;i=e[i].nex)
    {
        int v=e[i].v;
        if(v==pre) continue;
        dfs1(v,u,depth+1);
        siz[u]+=siz[v];
        if(siz[v]>siz[son[u]])
            son[u]=v;
    }
}
void dfs2(int u,int t) //处理 top id rk wt 数组
{
    top[u]=t;
    id[u]=++cnt;
    rk[cnt]=u;
    wt[cnt]=w[u];
    if(!son[u]) return;
    dfs2(son[u],t);
    for(int i=first[u];~i;i=e[i].nex)
    {
        int v=e[i].v;
        if(v!=son[u]&&v!=fa[u])
            dfs2(v,v);
    }
}
void pushup(int rt)
{
    sum[rt]=sum[rt<<1]+sum[rt<<1|1];
    maxx[rt]=max(maxx[rt<<1],maxx[rt<<1|1]);
}
void build(int l,int r,int rt)
{
    if(l==r)
    {
        sum[rt]=maxx[rt]=wt[l];
        return;
    }
    int mid=(l+r)>>1;
    build(lson);
    build(rson);
    pushup(rt);
}
void update(int k,int x,int l,int r,int rt)
{
    if(l==r)
    {
        sum[rt]=x;
        maxx[rt]=x;
        return;
    }
    int mid=(l+r)>>1;
    if(k<=mid)update(k,x,lson);
    else update(k,x,rson);
    pushup(rt);
}
int querymax(int L,int R,int l,int r,int rt)
{
    if(L<=l&&r<=R)
        return maxx[rt];
    int mid=(l+r)>>1,ans=-inf;
    if(L<=mid) ans=max(ans,querymax(L,R,lson));
    if(R>mid) ans=max(ans,querymax(L,R,rson));
    return ans;
}
int querysum(int L,int R,int l,int r,int rt)
{
    if(L<=l&&r<=R)
        return sum[rt];
    int mid=(l+r)>>1,ans=0;
    if(L<=mid) ans+=querysum(L,R,lson);
    if(R>mid) ans+=querysum(L,R,rson);
    return ans;
}

int quemax(int x,int y)
{
    int ans=-inf;
    while(top[x]!=top[y])
    {
        if(d[top[x]]<d[top[y]]) swap(x,y);
        ans=max(ans,querymax(id[top[x]],id[x],1,n,1));
        x=fa[top[x]];
    }
    if(d[x]>d[y]) swap(x,y);
    ans=max(ans,querymax(id[x],id[y],1,n,1));
    return ans;
}
int quesum(int x,int y)
{
    int ans=0;
    while(top[x]!=top[y])
    {
        if(d[top[x]]<d[top[y]]) swap(x,y);
        ans+=querysum(id[top[x]],id[x],1,n,1);
        x=fa[top[x]];
    }
    if(d[x]>d[y]) swap(x,y);
    ans+=querysum(id[x],id[y],1,n,1);
    return ans;
}
int main()
{
    scanf("%d",&n);
    init();
    int u,v,m;
    for(int i=1;i<n;i++)
    {
        scanf("%d%d",&u,&v);
        adde(u,v);
        adde(v,u);
    }
    for(int i=1;i<=n;i++)
        scanf("%d",&w[i]);
    dfs1(1,0,1);
    dfs2(1,1);
    build(1,n,1);
    char s[10];
    scanf("%d",&m);
    while(m--)
    {
        scanf("%s%d%d",s,&u,&v);
        if(s[0]=='C') update(id[u],v,1,n,1);
        else if(s[1]=='M')
        {
            int ans=quemax(u,v);
            printf("%d\n",ans);
        }
        else if(s[1]=='S')
        {
            int ans=quesum(u,v);
            printf("%d\n",ans);
        }
    }
    return 0;
}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值