Codevs 2460 == BZOJ 1036 树的统计
2460 树的统计
2008年省队选拔赛浙江
一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w。
我们将以下面的形式来要求你对这棵树完成一些操作:
- I. CHANGE u t : 把结点u的权值改为t
- II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值
- III. QSUM u v: 询问从点u到点v的路径上的节点的权值和
注意:从点u到点v的路径上的节点包括u和v本身
输入文件的第一行为一个整数n,表示节点的个数。
接下来n – 1行,每行2个整数a和b,表示节点a和节点b之间有一条边相连。
接下来n行,每行一个整数,第i行的整数wi表示节点i的权值。
接下来1行,为一个整数q,表示操作的总数。
接下来q行,每行一个操作,以“CHANGE u t”或者“QMAX u v”或者“QSUM u v”的形式给出。
对于每个“QMAX”或者“QSUM”的操作,每行输出一个整数表示要求输出的结果。
4
1 2
2 3
4 1
4 2 1 3
12
QMAX 3 4
QMAX 3 3
QMAX 3 2
QMAX 2 3
QSUM 3 4
QSUM 2 1
CHANGE 1 5
QMAX 3 4
CHANGE 3 6
QMAX 3 4
QMAX 2 4
QSUM 3 4
4
1
2
2
10
6
5
6
5
16
对于100%的数据,保证1<=n<=30000,0<=q<=200000;中途操作中保证每个节点的权值w在-30000到30000之间。
#include<iostream>
#include<cstdio>
#include<cstring>
#define maxn 300001
using namespace std;
int head[maxn],dep[maxn],fa[maxn],a[maxn];
int pos[maxn],top[maxn],size[maxn],son[maxn];
int tot,n,m,cnt,maxsize,x,y,num;
struct Edge{int next,to;}e[maxn<<];
struct Node{int l,r,max,sum;}tre[maxn<<];
int read(){
int x=,f=;char c=getchar();
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
}
int max(int a,int b){if(a>b)return a;else return b;}
void Add_Edge(int u,int v){
e[++num].to=v;e[num].next=head[u];head[u]=num;
}
void DFS_First(int now,int pre,int deepth){
dep[now] = deepth; fa[now] = pre; size[now] = ;
for(int i=head[now];i;i=e[i].next){
int v=e[i].to;
if(v!=pre){
DFS_First(v,now,deepth+);
size[now] += size[v];
if(son[now]==-||size[v]>size[son[now]])
son[now]=v;
}
}
}
void Get_Pos(int u,int Top){
int k=;maxsize++;
pos[u]=maxsize;top[u]=Top;
if(!son[u])return;
Get_Pos(son[u],Top);
for(int i=head[u];i;i=e[i].next)
if(dep[e[i].to]>dep[u]&&e[i].to!=son[u])
Get_Pos(e[i].to,e[i].to);
return;
}
void Build(int now,int l,int r){
tre[now].l=l;tre[now].r=r;
if(l==r){
tre[now].sum=tre[now].max=;return ;
}
int mid=(l+r)>>;
Build(now<<,l,mid);Build(now<<|,mid+,r);
tre[now].sum=tre[now<<].sum+tre[now<<|].sum;
tre[now].max=max(tre[now<<].max,tre[now<<|].max); }
void Insert(int now,int poss,int val){
if(tre[now].l==tre[now].r){
tre[now].sum=tre[now].max=val;return;
}
int mid=(tre[now].l+tre[now].r)/;
if(poss<=mid)Insert(now<<,poss,val);
else if(poss>mid)Insert(now<<|,poss,val);
tre[now].sum=tre[now<<].sum+tre[now<<|].sum;
tre[now].max=max(tre[now<<].max,tre[now<<|].max);
}
int QueryMax(int now,int l,int r){
if(l<=tre[now].l&&tre[now].r<=r)return tre[now].max;
int mid=(tre[now].l+tre[now].r)>>,total=-1e9;
if(l<=mid)total=max(total,QueryMax(now<<,l,r));
if(r>mid)total=max(total,QueryMax(now<<|,l,r));
return total;
}
int QuerySum(int now,int l,int r){
if(l<=tre[now].l&&tre[now].r<=r)return tre[now].sum;
int mid=(tre[now].l+tre[now].r)>>,total=;
if(l<=mid)total+=QuerySum(now<<,l,r);
if(r>mid)total+=QuerySum(now<<|,l,r);
return total;
}
int SolveMax(int u,int v){
int ans=-1e9;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]])swap(u,v);
ans=max(ans,QueryMax(,pos[top[u]],pos[u]));
u=fa[top[u]];
}
if(pos[u]>pos[v]) swap(u,v);
ans=max(ans,QueryMax(,pos[u],pos[v]));
return ans;
}
int SolveSum(int u,int v){
int ans=;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]])swap(u,v);
ans+=QuerySum(,pos[top[u]],pos[u]);
u=fa[top[u]];
}
if(pos[u]>pos[v])swap(u,v);
ans+=QuerySum(,pos[u],pos[v]);
return ans;
}
int main(){
n=read();
for(int i=;i<n;i++){
x=read();y=read();
Add_Edge(x,y);Add_Edge(y,x);
}
for(int i=;i<=n;i++) a[i]=read();
DFS_First(,,);
Get_Pos(,);
Build(,,maxsize);
for(int i=;i<=n;i++)
Insert(,pos[i],a[i]);
m=read();char sp[];
while(m--){
scanf("%s",sp);scanf("%d%d",&x,&y);
if(sp[]=='C')a[x]=y,Insert(,pos[x],y);
else{
if(sp[]=='M') printf("%d\n",SolveMax(x,y));
else printf("%d\n",SolveSum(x,y));
}
}
return ;
}
Codevs 2460 == BZOJ 1036 树的统计的更多相关文章
- BZOJ 1036 树的统计-树链剖分
[ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 12904 Solved: 5191[Submit][Status ...
- BZOJ 1036 树的统计
Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. Q ...
- BZOJ 1036 树的统计Count 树链剖分模板题
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1036 题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将 ...
- [置顶] bzoj 1036 树的统计Count 点权值模板
树链剖分 点权型可做模板,链路剖分的思想把点hash到线段树的上,然后可通过n*(log(n)*log(n))的复杂度在树上操作,在线段树上能操作的在链路上都能操作. #include<cstd ...
- BZOJ 1036 树的统计(树链剖分)
点权树链剖分模板题. # include <cstdio> # include <cstring> # include <cstdlib> # include &l ...
- bzoj 1036 树的统计Count
题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u ...
- BZOJ - 1036 树的统计Count (LCT)
LCT试炼题(代码量居然完爆树剖?) #include<bits/stdc++.h> using namespace std; ,inf=0x3f3f3f3f; ],flp[N],n,m, ...
- Bzoj 1036 树的统计 分类: ACM TYPE 2014-12-29 18:55 72人阅读 评论(0) 收藏
Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. Q ...
- BZOJ 1036 树的统计 | 树链剖分模板题
又做了一遍--去掉读入优化只有八十行~ #include <cstdio> #include <cstring> #include <algorithm> usin ...
随机推荐
- zabbix监控系统时间的问题
分类: 监控 2013-03-19 21:40:11 发现zabbix监控系统时间的一个问题!zabbix监控系统时间用的key是system.localtime,返回当前的系统时间,而配置tig ...
- LOJ#6342. 跳一跳(期望)
题意 $n \leqslant 10^5$ Sol 随便推一推就好了吧.. $f[i] = \frac{f[i] + f[i +1] + \dots f[n]}{n - i + 1} + 1$ 移一下 ...
- Mybatis 插入一条或批量插入 返回带有自增长主键记录
首先讲一下, 插入一条记录返回主键的 Mybatis 版本要求低点,而批量插入返回带主键的 需要升级到3.3.1版本,3.3.0之前的都不行, <dependency> <grou ...
- JavaScript常用八种继承方案
更新:在常用七种继承方案的基础之上增加了ES6的类继承,所以现在变成八种啦,欢迎加高级前端进阶群一起学习(文末). --- 2018.10.30 1.原型链继承 构造函数.原型和实例之间的关系:每个构 ...
- (转)CocoaPods
本文转自http://nshipster.cn/cocoapods/ 文明是建立在道路,桥梁,运河,下水道,管线,电线和光纤这些基础设施之上的.只要设计和施工得当,它们可以帮助社会成倍的发展. 唯一的 ...
- POJ:2185-Milking Grid(KMP找矩阵循环节)
Milking Grid Time Limit: 3000MS Memory Limit: 65536K Description Every morning when they are milked, ...
- 批量导出ppt中内嵌的图片
某个ppt中很多页,然后插入了很多图片,且图片都是被压缩的,看起来非常费劲,所以想着一次性把图片另存为,找了接近一个小时,终于被我找到啦,分享给大家: 1.直接把ppt的后缀修改为rar 2.解压ra ...
- Spring---环境搭建与包介绍
jar包下载路径 首先需要下载Spring框架 spring-framework-5.0.0.RELEASE-dist,官方地址为https://repo.spring.io/release/org/ ...
- S变换
哈哈,这两天在整理时频分析的方法,大部分参考网上写的比较好的资料,浅显易懂,在这谢过各位大神了! 今天准备写下S变换,由于网上资料较少,自己尝试总结下,学的不好,望各位多多指导 由前面的文章可知,傅里 ...
- 平衡树 - Luogu 1486 郁闷的出纳员
这么久没写平衡树了,再来一发... P1486 郁闷的出纳员 题目描述 OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的 ...