首页 > 其他 > 详细

hdu--3966(树链剖分)

时间:2015-04-19 17:59:15      阅读:156      评论:0      收藏:0      [点我收藏+]

题目链接:hdu--3966

给出n个点的值,还有n-1条边的连接方式,三种操作:

1、I在节点a到b的路径中所有的点都增加x

2、D在节点a到b的路径中所有的点都减少x

3、Q询问第k个节点的值。

将每个节点的值转化为父节点到子节点的边的权值,对于根节点做一个虚拟的父节点0 。进行树链剖分,整合到线段树中之后注意:

更新时,不能只更新a到b上的边的权值,因为那样会使b节点的权值不能被更新到,在更新a到b的段的权值后,应该在更新b到b的权值。

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std ;
#define INF 0x3f3f3f3f
#define int_now int l,int r,int rt
#define now l,r,rt
#define lson l,(l+r)/2,rt<<1
#define rson (l+r)/2+1,r,rt<<1|1
#define maxn 100000
#pragma comment(linker, "/STACK:1024000000,1024000000")
struct node
{
    int u , v ;
    int next ;
} edge[maxn<<1];
int head[maxn] , cnt , vis[maxn] ;
int num[maxn] , dep[maxn] , fa[maxn] , son[maxn] , top[maxn] , w[maxn] , step ;
int cl[maxn<<2] , lazy[maxn<<2] ;
int c[maxn] , belong[maxn] ;
void add(int u,int v)
{
    edge[cnt].u = u ;
    edge[cnt].v = v ;
    edge[cnt].next = head[u] ;
    head[u] = cnt++ ;
    return ;
}
void dfs1(int u)
{
    num[u] = 1 ;
    son[u] = -1 ;
    int i , v ;
    for(i = head[u] ; i != -1 ; i = edge[i].next)
    {
        v = edge[i].v ;
        if( vis[v] ) continue ;
        vis[v] = 1 ;
        dep[v] = dep[u] + 1 ;
        fa[v] = u ;
        dfs1(v) ;
        num[u] += num[v] ;
        if( son[u] == -1 || num[ son[u] ] < num[v] )
            son[u] = v ;
    }
    return ;
}
void dfs2(int u)
{
    if( son[u] == -1 ) return ;
    w[ son[u] ] = step++ ;
    top[ son[u] ] = top[u] ;
    vis[ son[u] ] = 1 ;
    dfs2(son[u]) ;
    int i , v ;
    for(i = head[u] ; i != -1 ; i = edge[i].next)
    {
        v = edge[i].v ;
        if( vis[v] ) continue ;
        vis[v] = 1 ;
        w[v] = step++ ;
        top[v] = v ;
        dfs2(v) ;
    }
    return ;
}
void dfs()
{
    memset(vis,0,sizeof(vis)) ;
    vis[0] = 1 ;
    dep[0] = 1 ;
    fa[0] = -1 ;
    dfs1(0) ;
    memset(vis,0,sizeof(vis)) ;
    vis[0] = 1 ;
    top[0] = 0 ;
    step = 1 ;
    dfs2(0) ;
    return ;
}
void build(int_now)
{
    cl[rt] = lazy[rt] = 0 ;
    if( l == r )
    {
        cl[rt] = c[ belong[l] ] ;
        return ;
    }
    build(lson) ;
    build(rson) ;
    return ;
}
void update(int ll,int rr,int_now,int x)
{
    if( l > rr || r < ll ) return ;
    if( ll <= l && rr >= r )
    {
        lazy[rt] += x ;
        return ;
    }
    update(ll,rr,lson,x) ;
    update(ll,rr,rson,x) ;
}
int query(int k,int_now,int sum)
{
    sum += lazy[rt] ;
    if( l == k && r == k )
    {
        return cl[rt] + sum ;
    }
    if( k <= (l+r)/2 )
        return query(k,lson,sum) ;
    else
        return query(k,rson,sum) ;
}
void solve(int u,int v,int s)
{
    int f1 , f2 ;
    while( u != v )
    {
        if( dep[u] > dep[v] )
            swap(u,v) ;
        f1 = top[u] ;
        f2 = top[v] ;
        if( f1 == f2 )
        {
            update(w[son[u]],w[v],1,step-1,1,s) ;
            v = u ;
        }
        else if( dep[f1] > dep[f2] )
        {
            update(w[f1],w[u],1,step-1,1,s) ;
            u = fa[f1] ;
        }
        else
        {
            update(w[f2],w[v],1,step-1,1,s) ;
            v = fa[f2] ;
        }
    }
    update(w[u],w[u],1,step-1,1,s) ;
    return ;
}
char str[10] ;
int main()
{
    int n , m , p ;
    int i , j  , k ;
    int u , v , s ;
    while( scanf("%d %d %d", &n, &m, &p) != EOF )
    {
        memset(head,-1,sizeof(head)) ;
        cnt = 0 ;
        for(i = 1 ; i <= n ; i++)
            scanf("%d", &c[i]) ;
        add(0,1) , add(1,0) ;
        while( m-- )
        {
            scanf("%d %d", &u, &v) ;
            add(u,v) ;
            add(v,u) ;
        }
        dfs() ;
        for(i = 1 ; i <= n ; i++)
            belong[ w[i] ] = i ;
        build(1,step-1,1) ;
        while( p-- )
        {
            scanf("%s", str) ;
            if( str[0] == 'I' )
            {
                scanf("%d %d %d", &u, &v, &s) ;
                solve(u,v,s) ;
            }
            else if( str[0] == 'D' )
            {
                scanf("%d %d %d", &u, &v, &s) ;
                solve(u,v,-s) ;
            }
            else
            {
                scanf("%d", &i) ;
                printf("%d\n", query(w[i],1,step-1,1,0)) ;
            }
        }
    }
    return 0 ;
}


 

hdu--3966(树链剖分)

原文:http://blog.csdn.net/winddreams/article/details/45130265

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!