您的位置:首页 > 其它

BZOJ4372: 烁烁的游戏

2015-12-30 21:00 323 查看

Description

背景:烁烁很喜欢爬树,这吓坏了树上的皮皮鼠。
题意:
给定一颗n个节点的树,边权均为1,初始树上没有皮皮鼠。
烁烁他每次会跳到一个节点u,把周围与他距离不超过d的节点各吸引出w只皮皮鼠。皮皮鼠会被烁烁吸引,所以会一直待在节点上不动。
烁烁很好奇,在当前时刻,节点u有多少个他的好朋友---皮皮鼠。
大意:
给一颗n个节点的树,边权均为1,初始点权均为0,m次操作:
Q x:询问x的点权。
M x d w:将树上与节点x距离不超过d的节点的点权均加上w。

Input

第一行两个正整数:n,m
接下来的n-1行,每行三个正整数u,v,代表u,v之间有一条边。
接下来的m行,每行给出上述两种操作中的一种。

Output

对于每个Q操作,输出当前x节点的皮皮鼠数量。

Sample Input

7 6

1 2

1 4

1 5

2 3

2 7

5 6

M 1 1 2

Q 5

M 2 2 3

Q 3

M 1 2 1

Q 2

Sample Output

2

3

6

HINT

数据范围:

n,m<=10^5,|w|<=10^4

注意:w不一定为正整数,因为烁烁可能把皮皮鼠吓傻了。

裸的动态树分治,和震波很像啊(时限宽松多了)。
套个支持区间增加的线段树就行了。
数组开小结果RE2次(TAT)。
前面10s的太强了。

#include<cstdio>
#include<cctype>
#include<queue>
#include<cmath>
#include<cstring>
#include<algorithm>
#define rep(i,s,t) for(int i=s;i<=t;i++)
#define dwn(i,s,t) for(int i=s;i>=t;i--)
#define ren for(int i=first[x];i;i=next[i])
using namespace std;
inline int read() {
int x=0,f=1;char c=getchar();
for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
for(;isdigit(c);c=getchar()) x=x*10+c-'0';
return x*f;
}
const int maxn=200010;
const int maxnode=20000010;
int n,m,first[maxn],next[maxn<<1],to[maxn<<1],e;
void AddEdge(int u,int v) {
to[++e]=v;next[e]=first[u];first[u]=e;
to[++e]=u;next[e]=first[v];first[v]=e;
}
int val[maxn],mn[maxn<<1][20],Log[maxn<<1],dep[maxn],pos[maxn],cnt;
void dfs(int x,int fa) {
dep[x]=dep[fa]+1;pos[x]=++cnt;mn[cnt][0]=dep[x];
ren if(to[i]!=fa) dfs(to[i],x),mn[++cnt][0]=dep[x];
}
void init() {
Log[0]=-1;
rep(i,1,cnt) Log[i]=Log[i>>1]+1;
for(int j=1;(1<<j)<=cnt;j++)
for(int i=1;i+(1<<j)-1<=cnt;i++)
mn[i][j]=min(mn[i][j-1],mn[i+(1<<j-1)][j-1]);
}
int dist(int x,int y) {
int k,ans=dep[x]+dep[y];
x=pos[x];y=pos[y];if(x>y) swap(x,y);k=Log[y-x+1];
return ans-2*min(mn[x][k],mn[y-(1<<k)+1][k]);
}
int vis[maxn],f[maxn],s[maxn],root,size;
void getroot(int x,int fa) {
int maxs=0;s[x]=1;
ren if(to[i]!=fa&&!vis[to[i]]) {
getroot(to[i],x);s[x]+=s[to[i]];
maxs=max(maxs,s[to[i]]);
}
f[x]=max(maxs,size-s[x]);
if(f[root]>f[x]) root=x;
}
int fa[maxn];
void build(int x) {
vis[x]=1;
ren if(!vis[to[i]]) {
size=f[0]=s[to[i]];getroot(to[i],root=0);
fa[root]=x;build(root);
}
}
int ls[maxnode],rs[maxnode],addv[maxnode],ToT;
int root1[maxn],root2[maxn];
void update(int& o,int l,int r,int ql,int qr,int v) {
if(!o) o=++ToT;
if(ql<=l&&r<=qr) addv[o]+=v;
else {
int mid=l+r>>1;
if(ql<=mid) update(ls[o],l,mid,ql,qr,v);
if(qr>mid) update(rs[o],mid+1,r,ql,qr,v);
}
}
int query(int& o,int l,int r,int pos,int cur) {
cur+=addv[o];
if(!o||pos<0||l==r) return cur;
int mid=l+r>>1;
if(pos<=mid) return query(ls[o],l,mid,pos,cur);
return query(rs[o],mid+1,r,pos,cur);
}
void update(int x,int w,int v) {
update(root1[x],0,n,0,w,v);
for(int i=x;fa[i];i=fa[i]) {
int D=dist(x,fa[i]);
if(w>=D) {
update(root1[fa[i]],0,n,0,w-D,v);
update(root2[i],0,n,0,w-D,v);
}
}
}
int query(int x) {
int res=query(root1[x],0,n,0,0);
for(int i=x;fa[i];i=fa[i]) {
int D=dist(x,fa[i]);
res+=query(root1[fa[i]],0,n,D,0)-query(root2[i],0,n,D,0);
}
return res;
}
int main() {
n=read();m=read();
rep(i,2,n) AddEdge(read(),read());
dfs(1,0);init();
f[0]=size=n;getroot(1,root=0);
build(root);
while(m--) {
char cmd[2];
scanf("%s",cmd);
if(cmd[0]=='Q') printf("%d\n",query(read()));
else {
int x=read(),w=read(),v=read();
update(x,w,v);
}
}
return 0;
}


View Code
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: