BZOJ 4034: [HAOI2015]T2

BZOJ 4034: [HAOI2015]T2

标签(空格分隔): OI BZOJ 树链剖分 线段树


Time Limit: 10 Sec
Memory Limit: 256 MB


Description

有一棵点数为 N 的树,以点 1 为根,且树点有边权。然后有 M 个

操作,分为三种:
操作 1 :把某个节点 x 的点权增加 a 。
操作 2 :把某个节点 x 为根的子树中所有点的点权都增加 a 。
操作 3 :询问某个节点 x 到根的路径中所有点的点权和。
Input

第一行包含两个整数 N, M 。表示点数和操作数。

接下来一行 N 个整数,表示树中节点的初始权值。
接下来 N-1 行每行三个正整数 fr, to , 表示该树中存在一条边 (fr, to) 。
再接下来 M 行,每行分别表示一次操作。其中第一个数表示该操
作的种类( 1-3 ) ,之后接这个操作的参数( x 或者 x a ) 。
Output

对于每个询问操作,输出该询问的答案。答案之间用换行隔开。

Sample Input

5 5

1 2 3 4 5

1 2

1 4

2 3

2 5

3 3

1 2 1

3 5

2 1 2

3 3
Sample Output

6

9

13

HINT

对于 100% 的数据, N,M<=100000 ,且所有输入数据的绝对值都不

会超过 10^6 。
Source

鸣谢bhiaibogf提供


Solution####

树链剖分+线段树


Code####

#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<algorithm>
#include<queue>
#include<set>
#include<map>
#include<bitset>
#include<vector>
using namespace std;
#define PA pair<int,int>
const int N=0,M=0;
int read()
{int s=0,f=1;char ch=getchar();
 while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
 while(ch>='0'&&ch<='9'){s=(s<<1)+(s<<3)+ch-'0';ch=getchar();}
 return s*f;
}
//smile please
int n,m;
int val[100005];
int lpos[100005],rpos[100005],up[100005],dfn,siz[100005],fa[100005];
long long sum[1<<18],tag[1<<18];
int be[100005],bn[200005],bv[200005],bw=1;
void put(int u,int v)
{bw++;bn[bw]=be[u];be[u]=bw;bv[bw]=v;}
int ss;
void dfs1(int x)
{
	for(int i=be[x];i;i=bn[i])
	   if(bv[i]!=fa[x])
	     fa[bv[i]]=x,
		 dfs1(bv[i]),
		 siz[x]+=siz[bv[i]];
	siz[x]++;
}
void dfs2(int x,int UP)
{
	int ma=0;
	for(int i=be[x];i;i=bn[i])
	   if(siz[bv[i]]>siz[ma]&&bv[i]!=fa[x])
	     ma=bv[i];
	up[x]=UP;lpos[x]=rpos[x]=++dfn;
	if(!ma)return;
	   dfs2(ma,UP);
	   rpos[x]=max(rpos[x],rpos[ma]);
	for(int i=be[x];i;i=bn[i])
	   {int v=bv[i];if(v==ma||v==fa[x])continue;
	    dfs2(v,v);
	    rpos[x]=max(rpos[x],rpos[v]);
	   }
}
void down(int l,int r,int x)
{
	if(l==r)return;
	long long t=tag[x];tag[x]=0;int mid=l+r>>1;
	tag[x<<1]+=t,sum[x<<1]+=t*(mid-l+1);
	tag[x<<1|1]+=t,sum[x<<1|1]+=t*(r-mid);
}
void add(int x,int l,int r,int L,int R,long long s)
{
	if(L>R)return;
	if(tag[x])down(l,r,x);
	if(l==L&&r==R)
	  {tag[x]+=s;sum[x]+=(r-l+1)*s;return;}
	int mid=l+r>>1;
	add(x<<1,l,mid,L,min(R,mid),s);
	add(x<<1|1,mid+1,r,max(L,mid+1),R,s);
	sum[x]=sum[x<<1]+sum[x<<1|1];
}
long long ask(int x,int l,int r,int L,int R)
{
	if(L>R)return 0;
	if(tag[x])down(l,r,x);
	if(l==L&&r==R)
	  return sum[x];
	int mid=l+r>>1;
	return ask(x<<1,l,mid,L,min(R,mid))+ask(x<<1|1,mid+1,r,max(L,mid+1),R);
}
long long ask1(int x)
{
	long long ans=0;
	for(;x;x=fa[up[x]])
	    ans+=ask(1,1,n,lpos[up[x]],lpos[x]);
	return ans;
}
int main()
{
	n=read(),m=read();
	for(int i=1;i<=n;i++)
	   val[i]=read();
	for(int i=1;i<n;i++)
	   {int f=read(),s=read();
	    put(s,f);
	    put(f,s);
	   }
	dfs1(1);
	dfs2(1,1);
	for(int i=1;i<=n;i++)
	   add(1,1,n,lpos[i],lpos[i],val[i]);
	while(m--)
	  {int k=read(),x=read();
	   if(k==1)add(1,1,n,lpos[x],lpos[x],read());
	   if(k==2)add(1,1,n,lpos[x],rpos[x],read());
	   if(k==3)printf("%lld\n",ask1(x));
	  }
	return 0;
}

posted on 2016-03-03 15:12  wuyuhan  阅读(202)  评论(0编辑  收藏  举报

导航