树链剖分

stO 树链剖分 Orz \color{red}\text{stO} \color{orange}\text{树链剖分}\color{green}\text{Orz} stO树链剖分Orz

题目描述
如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作:
操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z
操作2: 格式: 2 x y 表示求树从x到y结点最短路径上所有节点的值之和
操作3: 格式: 3 x z 表示将以x为根节点的子树内所有节点值都加上z
操作4: 格式: 4 x 表示求以x为根节点的子树内所有节点值之和
输入输出格式
输入格式:
第一行包含4个正整数N、M、R、P,分别表示树的结点个数、操作个数、根节点序号和取模数(即所有的输出结果均对此取模)。
接下来一行包含N个非负整数,分别依次表示各个节点上初始的数值。
接下来N-1行每行包含两个整数x、y,表示点x和点y之间连有一条边(保证无环且连通)
接下来M行每行包含若干个正整数,每行表示一个操作,格式如下:
操作1: 1 x y z
操作2: 2 x y
操作3: 3 x z
操作4: 4 x
输出格式:
输出包含若干行,分别依次表示每个操作2或操作4所得的结果(对P取模)
输入输出样例
输入样例#1:
5 5 2 24
7 3 7 8 0
1 2
1 5
3 1
4 1
3 4 2
3 2 2
4 5
1 5 1 3
2 1 3
输出样例#1:
2
21
说明
时空限制:1s,128M
数据规模:
对于30%的数据: N \leq 10, M \leq 10 N≤10,M≤10
对于70%的数据: N \leq {10}^3, M \leq {10}^3 N≤10
3
,M≤10
3
对于100%的数据: N \leq {10}^5, M \leq {10}^5 N≤10
5
,M≤10
5
( 其实,纯随机生成的树LCA+暴力是能过的,可是,你觉得可能是纯随机的么233 )
样例说明:
树的结构如下
各个操作如下:
故输出应依次为2、21(重要的事情说三遍:记得取模)

// luogu-judger-enable-o2
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
#define maxn 444444
using namespace std ;
struct dy{
	int x , y , z , next ;
}a[maxn] ;
struct Tree {
	int lc,rc,sum,tag ;
}tree[maxn] ;
int deep[maxn] , top[maxn] , id[maxn] , size[maxn] ,son[maxn] ;
int n , m , r , p ;
int tot , cnt = 1 ;
int head[maxn] , fa[maxn] , w[maxn] ,wt[maxn] , tt ;
void pushup(int u) {
	tree[u].sum = (tree[tree[u].lc].sum + tree[tree[u].rc].sum)%p ;
	return ;
} 
void pushdown(int u ,int l ,int r) {
	int mid = (l+r) >> 1 ;
	tree[tree[u].lc].sum = (tree[tree[u].lc].sum+(mid-l+1)*tree[u].tag) % p ;
	tree[tree[u].lc].tag = (tree[u].tag+tree[tree[u].lc].tag) % p ;
	tree[tree[u].rc].sum = (tree[tree[u].rc].sum + tree[u].tag*(r-mid))%p ;
	tree[tree[u].rc].tag = (tree[tree[u].rc].tag+tree[u].tag)%p ;
	tree[u].tag = 0 ;
	return ;
}
void build(int u ,int l ,int r) {
	if(l == r) {
		tree[u].sum = wt[l] ;
		return ;
	}
	int mid = (l+r) >> 1 ;
	tree[u].lc = ++cnt ;
	build(tree[u].lc,l,mid) ;
	tree[u].rc = ++cnt ;
	build(tree[u].rc , mid+1,r) ;
	pushup(u) ;
}
int query(int u ,int l ,int r,int ll,int rr) {
	if(l == ll && r == rr) {
		return tree[u].sum ;
	}int mid = (l+r) >>1 ;
	pushdown(u,l,r) ;
	if(rr <= mid) return query(tree[u].lc,l,mid,ll,rr) ;
	else if(ll > mid) return query(tree[u].rc,mid+1,r,ll,rr) ;
	else {
		return (query(tree[u].lc,l,mid,ll,mid)+query(tree[u].rc,mid+1,r,mid+1,rr))%p ;
	}
	pushup(u) ;
}
void updata(int u ,int l ,int r,int ll ,int rr,int w) {
	if(l == ll &&r == rr) {
		tree[u].sum = (tree[u].sum + w*(r-l+1)) % p ;
		tree[u].tag = (tree[u].tag + w) % p ;
		return ;
	}int mid = (l+r) >> 1 ;
	pushdown(u,l,r) ;
	if(rr <= mid) updata(tree[u].lc,l,mid,ll,rr,w) ;
	else if(ll > mid) updata(tree[u].rc,mid+1,r,ll,rr,w) ;
	else {
		updata(tree[u].lc,l,mid,ll,mid,w) ;
		updata(tree[u].rc,mid+1,r,mid+1,rr,w) ;
	}
	pushup(u) ;
}
void dfs1(int x,int f,int depth) {
    size[x] = 1 ;
    fa[x] = f ;
    deep[x] = depth ;
    for(int i = head[x] ; i ; i = a[i].next) {
        int v = a[i].y ;
        if(v == f ) continue;
        dfs1(v,x,depth+1);
        size[x] += size[v];
        if(size[v] > size[son[x]]) {
            son[x] = v ;
        }
    }
}
void dfs2(int x,int topf) {
    id[x] = ++ tt ;
    wt[tt] = w[x];
    top[x] = topf;
    if(!son[x]) return;
    dfs2(son[x],topf);
    for(int i = head[x] ; i ; i = a[i].next) {
        int v = a[i].y;
        if(v == son[x] || v == fa[x]) continue;
        dfs2(v,v);
    }
}
int qlink(int x,int y) {
    int ans = 0 ;
    while(top[x] != top[y]) {
        if(deep[top[x]] < deep[top[y]]) swap(x,y);
        ans += query(1,1,n,id[top[x]],id[x]);
        ans%=p ;
        x = fa[top[x]];
    }
    if(deep[x] > deep[y]) swap(x,y) ;
    ans = (ans + query(1,1,n,id[x],id[y]))%p;
    return ans;
}
void uplink(int x,int y,int w) {
    while(top[x] != top[y]) {
        if(deep[top[x]] < deep[top[y]]) swap(x,y);
        updata(1,1,n,id[top[x]],id[x],w);
        x = fa[top[x]];
    }
    if(deep[x] > deep[y]) swap(x,y);
    updata(1,1,n,id[x],id[y],w);
}
int qson(int x) {
    return query(1,1,n,id[x],id[x]+size[x]-1);
}
void upson(int x,int w) {
    updata(1,1,n,id[x],id[x]+size[x]-1,w);
}
void add(int x ,int y) {
	a[++tot].x = x ;
	a[tot].y = y ;
	a[tot].next = head[x] ;
	head[x] = tot ;
}
int main () {
	cin >> n >> m >> r >> p ;
    for(int i = 1 ; i <= n ; i ++)
        cin >> w[i] ;
    for(int i = 1 ; i < n ; i ++) {
        int x,y;
       	cin >> x >> y ;
        add(x,y);
        add(y,x);
    }
    dfs1(r,0,1);
    dfs2(r,r);
    build(1,1,n);
    while(m--) {
        int opt;
        cin >> opt ;
        if(opt == 1) {
            int x,y,z;
            cin >> x >> y >> z ;
            uplink(x,y,z);
        }
        if(opt == 2) {
            int x,y;
            cin >> x >> y ;
            printf("%d\n",qlink(x,y)) ;
        }
        if(opt == 3) {
            int x,z;
           	cin >> x >> z ;
            upson(x,z) ;
        }
        if(opt == 4) {
            int x ;
            cin >> x ;
            printf("%d\n",qson(x));
        }
    }
	return 0 ;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值