HDU 5044 Tree 树链剖分

一棵树,初始边权和点权都为0

现在有m个操作,每一个操作:

ADD1 u v k: for nodes on the path from u to v, the value of these nodes increase by k. 

ADD2 u v k: for edges on the path from u to v, the value of these edges increase by k. 

 

操作完后,输出每一个点的点权和每一条边的边权(边权按照输入顺序输出)

 

我们把边权也当做点权处理

这道题本来是一道痕裸的树链剖分的题目,用树状数组维护就可以了O(nlogn*logn)

不过这样会T,因为出题人特意卡了常数

有人是加了一个输入输出外挂过的

 

不过这道题有更好的做法

 

我们在树链剖分后的数组用标记法,每次对一个连续区间+k时,我们对位置L标记+k,对位置R+1标记-k

最后从前往后加一遍就是了

O(nlogn)

 

注意:处理边权和点权时,对lca的处理是不同的

点权时,lca的点权也要更新

边权时,lca对应的边的边权时不需要更新的

 

注意2:当节点个数为1时,是没有边的,也就没有边权了,但是输出的时候我们边权这一行依然要输出,是一个空行

 

 

 

 

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#include<stack>
#include<map>
#include<set>

#define LL long long
#define pb push_back
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1

using namespace std;

const int maxn=1e5+10;

int dep[maxn];
int siz[maxn];
int chg[maxn];
int fa[maxn];
int son[maxn];
int top[maxn];
LL c[2][maxn];
int n;
int e[maxn][2];

struct Edge
{
    int to,next;
};
Edge edge[maxn<<1];
int head[maxn];
int tot;

void init()
{
    memset(head,-1,sizeof head);
    tot=0;
}

void addedge(int u,int v)
{
    edge[tot].to=v;
    edge[tot].next=head[u];
    head[u]=tot++;
}

void solve(int );

int main()
{
    int test;
    int cas=1;
    scanf("%d",&test);
    while(test--){
        printf("Case #%d:\n",cas++);
        init();
        int q;
        scanf("%d %d",&n,&q);
        for(int i=1;i<n;i++){
            scanf("%d %d",&e[i][0],&e[i][1]);
            addedge(e[i][0],e[i][1]);
            addedge(e[i][1],e[i][0]);
        }
        solve(q);
    }
    return 0;
}

void dfs0(int u,int pre)
{
    dep[u]=dep[pre]+1;
    fa[u]=pre;
    siz[u]=1;
    for(int i=head[u];~i;i=edge[i].next){
        int v=edge[i].to;
        if(v==pre)
            continue;
        dfs0(v,u);
        siz[u]+=siz[v];
        if(son[u]==-1 || siz[v]>siz[son[u]])
            son[u]=v;
    }
}

void dfs1(int u,int tp)
{
    top[u]=tp;
    chg[u]=++tot;
    if(son[u]==-1)
        return ;
    dfs1(son[u],tp);
    for(int i=head[u];~i;i=edge[i].next){
        int v=edge[i].to;
        if(v==fa[u] || v==son[u])
            continue;
        dfs1(v,v);
    }
}

void update_path(int a,int b,int cnt,int k)
{
    while(top[a]!=top[b]){
        if(dep[top[a]]<dep[top[b]])
            swap(a,b);
        c[cnt][chg[top[a]]]+=k;
        c[cnt][chg[a]+1]-=k;
        a=fa[top[a]];
    }
    if(dep[a]<dep[b])
        swap(a,b);
    if(cnt==1){
        c[cnt][chg[b]]+=k;
        c[cnt][chg[a]+1]-=k;
    }
    else{
        c[cnt][chg[b]+1]+=k;
        c[cnt][chg[a]+1]-=k;
    }
    return ;
}

void solve(int q)
{
    memset(c,0,sizeof c);
    memset(dep,0,sizeof dep);
    memset(son,-1,sizeof son);

    dfs0(1,1);
    tot=0;
    dfs1(1,1);

    for(int i=1;i<=q;i++){
        char str[10];
        scanf("%s",str);
        int u,v,k;
        scanf("%d %d %d",&u,&v,&k);
        if(str[3]=='1'){
            update_path(u,v,1,k);
        }
        else{
            update_path(u,v,0,k);
        }
    }

    for(int i=1;i<=n;i++){
        c[0][i]+=c[0][i-1];
        c[1][i]+=c[1][i-1];
    }
    for(int i=1;i<=n;i++){
        printf("%d",c[1][chg[i]]);
        if(i<n)
            printf(" ");
        else
            puts("");
    }
    for(int i=1;i<n;i++){
        int cur=e[i][0];
        if(dep[e[i][0]]<dep[e[i][1]])
            cur=e[i][1];
        printf("%d",c[0][chg[cur]]);
        if(i<n-1)
            printf(" ");
    }
    puts("");
    return ;
}

 

posted on 2015-10-10 19:46  _fukua  阅读(247)  评论(0编辑  收藏  举报