您的位置:首页 > 其它

[BZOJ1984] 月下“毛景树”|树链剖分|线段树

2015-08-11 19:25 344 查看

1984: 月下“毛景树”

Time Limit: 20 Sec Memory Limit: 64 MB
Submit: 1088 Solved: 348
[Submit][Status][Discuss]

Description

毛毛虫经过及时的变形,最终逃过的一劫,离开了菜妈的菜园。 毛毛虫经过千山万水,历尽千辛万苦,最后来到了小小的绍兴一中的校园里。爬啊爬~爬啊爬~~毛毛虫爬到了一颗小小的“毛景树”下面,发现树上长着他最爱吃的毛毛果~~~ “毛景树”上有N个节点和N-1条树枝,但节点上是没有毛毛果的,毛毛果都是长在树枝上的。但是这棵“毛景树”有着神奇的魔力,他能改变树枝上毛毛果的个数:  Change k w:将第k条树枝上毛毛果的个数改变为w个。  Cover u v w:将节点u与节点v之间的树枝上毛毛果的个数都改变为w个。  Add u v w:将节点u与节点v之间的树枝上毛毛果的个数都增加w个。 由于毛毛虫很贪,于是他会有如下询问:  Max u v:询问节点u与节点v之间树枝上毛毛果个数最多有多少个。

Input

第一行一个正整数N。 接下来N-1行,每行三个正整数Ui,Vi和Wi,第i+1行描述第i条树枝。表示第i条树枝连接节点Ui和节点Vi,树枝上有Wi个毛毛果。 接下来是操作和询问,以“Stop”结束。

Output

对于毛毛虫的每个询问操作,输出一个答案。

Sample Input

4

1 2 8

1 3 7

3 4 9

Max 2 4

Cover 2 4 5

Add 1 4 10

Change 1 16

Max 2 4

Stop

Sample Output

9

16

【Data Range】

1<=N<=100,000,操作+询问数目不超过100,000。

保证在任意时刻,所有树枝上毛毛果的个数都不会超过10^9个。

HINT

Source

树的分治

1071046ws_fqk1984Wrong_Answer44644 kb904 msC++/Edit5488 B2015-08-11 19:22:07
1071034ws_fqk1984Wrong_Answer44644 kb924 msC++/Edit5470 B2015-08-11 19:15:21
1071019ws_fqk1984Wrong_Answer42304 kb960 msC++/Edit5312 B2015-08-11 19:08:32
1071018ws_fqk1984Wrong_Answer43464 kb92 msC++/Edit4179 B2015-08-11 19:07:15
1071006ws_fqk1984Time_Limit_Exceed0 kb0 msC++/Edit5314 B2015-08-11 18:55:07
1071002ws_fqk1984Time_Limit_Exceed0 kb0 msC++/Edit5308 B2015-08-11 18:51:23
1070994ws_fqk1984Runtime_Error21988 kb976 msC++/Edit5308 B2015-08-11 18:42:40
已经给跪了……

大体思路就是把边权压到点上,具体做法可以自己YY,询问时处理一下就好。hzwer学长是压到了深度小的点,我是压到了深度大的点。

UPD:8.13 终于A掉了!

1072820ws_fqk1984Accepted23968 kb5604 msC++/Edit4935 B2015-08-13 11:00:58
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
#include<set>
#include<map>
#include<queue>
#define N 100005
using namespace std;
int head
,next[2*N],list[2*N],key[2*N];
int l[4*N],r[4*N],tagadd[4*N],tagchange[4*N],size
,mx[4*N],c
,fa
[20],deep
,id
,belong
;
int n,cnt,dfn;
struct node {int u,v,w;} e
;
char ch[10];
inline int read()
{
int a=0,f=1; char c=getchar();
while (c<'0'||c>'9') {if (c=='-') f=-1; c=getchar();}
while (c>='0'&&c<='9') {a=a*10+c-'0'; c=getchar();}
return a*f;
}
inline void insert(int x,int y,int z)
{
next[++cnt]=head[x];
head[x]=cnt;
list[cnt]=y;
key[cnt]=z;
}
void dfs1(int x)
{
size[x]=1;
for (int i=1;(1<<i)<=deep[x];i++) fa[x][i]=fa[fa[x][i-1]][i-1];
for (int i=head[x];i;i=next[i])
{
if (list[i]==fa[x][0]) continue;
fa[list[i]][0]=x;
c[list[i]]+=key[i];
deep[list[i]]=deep[x]+1;
dfs1(list[i]);
size[x]+=size[list[i]];
}
}
void dfs2(int x,int chain)
{
belong[x]=chain;
id[x]=++dfn;
int k=0;
for (int i=head[x];i;i=next[i])
if (list[i]!=fa[x][0]&&size[list[i]]>size[k]) k=list[i];
if (!k) return;
dfs2(k,chain);
for (int i=head[x];i;i=next[i])
if (list[i]!=fa[x][0]&&list[i]!=k) dfs2(list[i],list[i]);
}
inline int lca(int x,int y)
{
if (deep[x]<deep[y]) swap(x,y);
int t=deep[x]-deep[y];
for (int i=0;(1<<i)<=t;i++)
if ((1<<i)&t) x=fa[x][i];
for (int i=18;i>=0;i--)
if (fa[x][i]!=fa[y][i]) {x=fa[x][i]; y=fa[y][i];}
return x==y?x:fa[x][0];
}
void build(int k,int x,int y)
{
l[k]=x; r[k]=y; tagchange[k]=-1;
if (x==y) return;
int mid=(x+y)>>1;
build(k<<1,x,mid); build(k<<1|1,mid+1,y);
}
inline void pushup(int k)
{
mx[k]=max(mx[k<<1],mx[k<<1|1]);
}
inline void pushdown(int k)
{
if (l[k]==r[k]) return;
if (tagchange[k]!=-1)
{
tagadd[k<<1]=tagadd[k<<1|1]=0;
tagchange[k<<1]=tagchange[k<<1|1]=tagchange[k];
mx[k<<1]=mx[k<<1|1]=tagchange[k];
tagchange[k]=-1;
}
if (tagadd[k])
{
mx[k<<1]+=tagadd[k]; mx[k<<1|1]+=tagadd[k];
if (tagchange[k<<1]!=-1) tagchange[k<<1]+=tagadd[k]; else tagadd[k<<1]+=tagadd[k];
if (tagchange[k<<1|1]!=-1) tagchange[k<<1|1]+=tagadd[k]; else tagadd[k<<1|1]+=tagadd[k];
tagadd[k]=0;
}
}
void change(int k,int x,int y,int v)
{
pushdown(k);
if (l[k]==x&&r[k]==y)
{
tagchange[k]=mx[k]=v;
return;
}
int mid=(l[k]+r[k])>>1;
if (y<=mid) change(k<<1,x,y,v);
else if (x>mid) change(k<<1|1,x,y,v);
else
{
change(k<<1,x,mid,v);
change(k<<1|1,mid+1,y,v);
}
pushup(k);
}
void add(int k,int x,int y,int v)
{
pushdown(k);
if (l[k]==x&&r[k]==y)
{
tagadd[k]=v;
mx[k]+=v;
return;
}
int mid=(l[k]+r[k])>>1;
if (y<=mid) add(k<<1,x,y,v);
else if (x>mid) add(k<<1|1,x,y,v);
else
{
add(k<<1,x,mid,v);
add(k<<1|1,mid+1,y,v);
}
pushup(k);
}
int getmx(int k,int x,int y)
{
pushdown(k);
if (l[k]==x&&r[k]==y) return mx[k];
int mid=(l[k]+r[k])>>1;
if (mid>=y) return getmx(k<<1,x,y);
else if (mid<x)return getmx(k<<1|1,x,y);
else return max(getmx(k<<1,x,mid),getmx(k<<1|1,mid+1,y));
}
inline void solvechange(int x,int f,int w)
{
while (belong[x]!=belong[f])
{
change(1,id[belong[x]],id[x],w);
x=fa[belong[x]][0];
}
if (id[f]+1<=id[x]) change(1,id[f]+1,id[x],w);
}
inline int solvemax(int x,int f)
{
int ans=-1;
while (belong[x]!=belong[f])
{
ans=max(ans,getmx(1,id[belong[x]],id[x]));
x=fa[belong[x]][0];
}
if (id[f]+1<=id[x]) ans=max(ans,getmx(1,id[f]+1,id[x]));
return ans;
}
inline void solveadd(int x,int f,int w)
{
while (belong[x]!=belong[f])
{
add(1,id[belong[x]],id[x],w);
x=fa[belong[x]][0];
}
if (id[f]+1<=id[x]) add(1,id[f]+1,id[x],w);
}
int main()
{
n=read();
for (int i=1;i<n;i++)
{
e[i].u=read(),e[i].v=read(),e[i].w=read();
insert(e[i].u,e[i].v,e[i].w); insert(e[i].v,e[i].u,e[i].w);
}
dfs1(1); dfs2(1,1); build(1,1,n);
for (int i=1;i<=n;i++) change(1,id[i],id[i],c[i]);
while (1)
{
scanf("%s",ch);
if (ch[1]=='t') break;
int u=read(),v=read(),t,w;
if (ch[1]=='a') {t=lca(u,v); printf("%d\n",max(solvemax(u,t),solvemax(v,t)));}
if (ch[1]=='o') {w=read(); t=lca(u,v); solvechange(u,t,w); solvechange(v,t,w);}
if (ch[1]=='d') {w=read(); t=lca(u,v); solveadd(u,t,w); solveadd(v,t,w);}
if (ch[1]=='h') {if (deep[e[u].v]<deep[e[u].u]) change(1,id[e[u].u],id[e[u].u],v); else change(1,id[e[u].v],id[e[u].v],v);}
}
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: