zoukankan      html  css  js  c++  java
  • bzoj 1036 树链剖分

    1036: [ZJOI2008]树的统计Count

    Time Limit: 10 Sec  Memory Limit: 162 MB
    Submit: 18993  Solved: 7748
    [Submit][Status][Discuss]

    Description

      一棵树上有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

    HINT

     

    Source

     代码:
    //和上一题稍微不同,这题权值在点上。
    #include<iostream>
    #include<cstdio>
    #include<cstring>
    using namespace std;
    const int MAXN=30000;
    int fa[MAXN+9],son[MAXN+9],lev[MAXN+9],top[MAXN+9],id[MAXN+9],head[MAXN+9],size[MAXN+9],val[MAXN+9];
    int max_[MAXN*4+9],sum_[MAXN*4+9],tot,cnt;
    struct Edge
    {
        int u,v,next;
    }edge[MAXN*2+9];
    void init()
    {
        tot=cnt=0;
        for(int i=0;i<=MAXN;i++) fa[i]=top[i]=i;
        memset(sum_,0,sizeof(sum_));
        memset(head,-1,sizeof(head));
    }
    void add(int x,int y)
    {
        edge[tot].u=x;edge[tot].v=y;
        edge[tot].next=head[x];
        head[x]=tot++;
        edge[tot].u=y;edge[tot].v=x;
        edge[tot].next=head[y];
        head[y]=tot++;
    }
    void dfs1(int x,int d)
    {
        lev[x]=d;
        son[x]=0;
        size[x]=1;
        for(int i=head[x];i!=-1;i=edge[i].next){
            int y=edge[i].v;
            if(y==fa[x]) continue;
            fa[y]=x;
            dfs1(y,d+1);
            size[x]+=size[y];
            if(size[son[x]]<size[y]) son[x]=y;
        }
    }
    void dfs2(int x,int tp)
    {
        top[x]=tp;
        id[x]=++cnt;
        if(son[x]) dfs2(son[x],tp);
        for(int i=head[x];i!=-1;i=edge[i].next){
            int y=edge[i].v;
            if(y==fa[x]||y==son[x]) continue;
            dfs2(y,y);
        }
    }
    void pushup(int rt)
    {
        max_[rt]=max(max_[rt<<1],max_[rt<<1|1]);
        sum_[rt]=sum_[rt<<1]+sum_[rt<<1|1];
    }
    void build(int l,int r,int rt)
    {
        if(l==r){
            max_[rt]=sum_[rt]=val[l];
            return;
        }
        int mid=(l+r)>>1;
        build(l,mid,rt<<1);
        build(mid+1,r,rt<<1|1);
        pushup(rt);
    }
    void update(int id,int c,int l,int r,int rt)
    {
        if(l==r){
            max_[rt]=sum_[rt]=c;
            return;
        }
        int mid=(l+r)>>1;
        if(id<=mid) update(id,c,l,mid,rt<<1);
        else update(id,c,mid+1,r,rt<<1|1);
        pushup(rt);
    }
    int query1(int ql,int qr,int l,int r,int rt)
    {
        if(ql<=l&&qr>=r) return max_[rt];
        int mid=(l+r)>>1,ans=-MAXN-9;
        if(ql<=mid) ans=max(ans,query1(ql,qr,l,mid,rt<<1));
        if(qr>mid) ans=max(ans,query1(ql,qr,mid+1,r,rt<<1|1));
        return ans;
    }
    int query2(int ql,int qr,int l,int r,int rt)
    {
        if(ql<=l&&qr>=r) return sum_[rt];
        int mid=(l+r)>>1,ans=0;
        if(ql<=mid) ans+=query2(ql,qr,l,mid,rt<<1);
        if(qr>mid) ans+=query2(ql,qr,mid+1,r,rt<<1|1);
        return ans;
    }
    int solve(int l,int r,bool f)
    {
        int ltp=top[l],rtp=top[r],ans=(f?0:-MAXN-9);
        while(ltp!=rtp){
            if(lev[rtp]<lev[ltp]){
                swap(ltp,rtp);
                swap(l,r);
            }
            if(!f) ans=max(ans,query1(id[rtp],id[r],1,cnt,1));
            else ans+=query2(id[rtp],id[r],1,cnt,1);
            r=fa[rtp];
            rtp=top[r];
        }
        if(lev[r]>lev[l]) swap(l,r);
        if(!f) ans=max(ans,query1(id[r],id[l],1,cnt,1));
        else ans+=query2(id[r],id[l],1,cnt,1);
        return ans;
    }
    int main()
    {
        //freopen("in.txt","r",stdin);
        int n,q;
        scanf("%d",&n);
        init();
        for(int i=1;i<n;i++){
            int x,y;
            scanf("%d%d",&x,&y);
            add(x,y);
        }
        dfs1(1,1);
        dfs2(1,1);
        for(int i=1;i<=n;i++){
            int x;
            scanf("%d",&x);
            val[id[i]]=x;
        }
        build(1,cnt,1);
        scanf("%d",&q);
        char ch[10];
        while(q--){
            int x,y;
            scanf("%s%d%d",ch,&x,&y);
            if(ch[0]=='C') update(id[x],y,1,cnt,1);
            else if(ch[1]=='M') printf("%d
    ",solve(x,y,0));
            else printf("%d
    ",solve(x,y,1));
        }
        return 0;
    }
  • 相关阅读:
    常规渗透:没遇到过的anquan狗
    SQLi-db 批量注入工具+教程
    国外整理的一套在线渗透测试资源合集
    重大漏洞!PHP multipart/form-data头部解析远程拒绝服务漏洞
    MS15-051 修正版Exploit(Webshell可用)
    网站渗透常用到的Python小脚本
    项目<<魔兽登录系统>>
    第七章:存储过程
    第六章:事务,视图和索引
    SQL高级查询:嵌套和分页
  • 原文地址:https://www.cnblogs.com/--ZHIYUAN/p/7690697.html
Copyright © 2011-2022 走看看