BZOJ1036:[ZJOI2008]树的统计——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=1036
题目描述
一棵树上有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之间。
——————————————————
树链剖分模板题。
具体树链剖分的讲解请看:http://www.cnblogs.com/luyouqi233/p/7886709.html
//luogu2590
//ZJOI2008树的统计
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
const int N=;
const int INF=;
inline int read(){
int X=,w=;char ch=;
while(ch<''||ch>''){w|=ch=='-';ch=getchar();}
while(ch>=''&&ch<='')X=(X<<)+(X<<)+(ch^),ch=getchar();
return w?-X:X;
}
struct node{
int to;
int nxt;
}edge[*N];
int head[N],cnt=,n;
inline void add(int u,int v){
cnt++;
edge[cnt].to=v;
edge[cnt].nxt=head[u];
head[u]=cnt;
return;
}
int fa[N],dep[N],size[N],son[N],top[N],pos[N],idx[N];
//依次为u的父亲,深度,重量,重儿子,重路径顶端,映射,反映射
int val[N],sum[N*],maxn[N*];
//依次为u的点权,区间和,区间最大值
void dfs1(int u){//处理fa,dep,size,son
size[u]=;
for(int i=head[u];i;i=edge[i].nxt){
int v=edge[i].to;
if(v==fa[u])continue;
fa[v]=u;dep[v]=dep[u]+;
dfs1(v);
size[u]+=size[v];
if(!son[u]||size[v]>size[son[u]])son[u]=v;//计算重儿子
}
return;
}
int tot;
void dfs2(int u,int anc){//处理top,pos,idx
tot++;
pos[u]=tot;
idx[tot]=u;
top[u]=anc;
if(!son[u])return;//到叶子了
dfs2(son[u],anc);//重路径上的点要在一段连续区间内所以先走重儿子
for(int i=head[u];i;i=edge[i].nxt){
int v=edge[i].to;
if(v==fa[u]||v==son[u])continue;
dfs2(v,v);//轻链top(anc)为自己
}
return;
}
void build(int a,int l,int r){//线段树建树
if(l==r){
sum[a]=maxn[a]=val[idx[l]];
return;
}
int mid=(l+r)>>;
build(a*,l,mid);
build(a*+,mid+,r);
sum[a]=sum[a*]+sum[a*+];
maxn[a]=max(maxn[a*],maxn[a*+]);
return;
}
int querysum(int a,int l,int r,int l1,int r1){//线段树区间和
if(r1<l||l1>r)return ;
if(l1<=l&&r<=r1)return sum[a];
int mid=(l+r)>>;
return querysum(a*,l,mid,l1,r1)+querysum(a*+,mid+,r,l1,r1);
}
int querymax(int a,int l,int r,int l1,int r1){//线段树区间最大值
if(r1<l||l1>r)return -INF;
if(l1<=l&&r<=r1)return maxn[a];
int mid=(l+r)>>;
return max(querymax(a*,l,mid,l1,r1),querymax(a*+,mid+,r,l1,r1));
}
void modify(int a,int l,int r,int p,int v){//线段树改值
if(p<l||r<p)return;
if(l==r){
sum[a]=maxn[a]=v;
return;
}
int mid=(l+r)>>;
modify(a*,l,mid,p,v);
modify(a*+,mid+,r,p,v);
sum[a]=sum[a*]+sum[a*+];
maxn[a]=max(maxn[a*],maxn[a*+]);
return;
}
int pathsum(int u,int v){//询问(u,v)这条路径的和
if(top[u]!=top[v]){//不在同一条重链
if(dep[top[u]]<dep[top[v]]){int t=u;u=v;v=t;}//一次爬少些,防止爬太大从而搜点搜多了
return pathsum(fa[top[u]],v)+querysum(,,n,pos[top[u]],pos[u]);//爬掉一整个重路径
}
if(dep[u]>dep[v]){int t=u;u=v;v=t;}
return querysum(,,n,pos[u],pos[v]);//一条重路径上一段
//此时u是深度较小的那个点,也就是原路径的LCA
}
int pathmax(int u,int v){//询问(u,v)这条路径的最大值,代码含义基本同上
if(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]){int t=u;u=v;v=t;}
return max(pathmax(fa[top[u]],v),querymax(,,n,pos[top[u]],pos[u]));
}
if(dep[u]>dep[v]){int t=u;u=v;v=t;}
return querymax(,,n,pos[u],pos[v]);
}
void init(){//初始化
dep[]=fa[]=;
dfs1();
top[]=idx[]=pos[]=;
tot=;
dfs2(,);
return;
}
int main(){
n=read();
for(int i=;i<=n;i++){
int u=read();
int v=read();
add(u,v);
add(v,u);
}
for(int i=;i<=n;i++)val[i]=read();
init();
build(,,n);
int q=read();
while(q--){
char op[];
scanf("%s",op);
int u=read();
int v=read();
if(op[]=='C')modify(,,n,pos[u],v);
else if(op[]=='S')printf("%d\n",pathsum(u,v));
else printf("%d\n",pathmax(u,v));
}
return ;
}
BZOJ1036:[ZJOI2008]树的统计——题解的更多相关文章
- [BZOJ1036][ZJOI2008]树的统计Count 解题报告|树链剖分
树链剖分 简单来说就是数据结构在树上的应用.常用的为线段树splay等.(可现在splay还不会敲囧) 重链剖分: 将树上的边分成轻链和重链. 重边为每个节点到它子树最大的儿子的边,其余为轻边. 设( ...
- bzoj1036 [ZJOI2008]树的统计Count
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MB Submit: 12646 Solved: 5085 [Subm ...
- bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题
[ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...
- bzoj千题计划124:bzoj1036: [ZJOI2008]树的统计Count
http://www.lydsy.com/JudgeOnline/problem.php?id=1036 树链剖分板子题 #include<cstdio> #include<iost ...
- BZOJ1036[ZJOI2008]树的统计Count 题解
题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.有一些操作:1.把结点u的权值改为t:2.询问从点u到点v的路径上的节点的最大权值 3.询问从点u到点v的路径上的节点的权值和 ...
- BZOJ1036 [ZJOI2008]树的统计Count 树链剖分
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1036 题意概括 一个树,每个节点有一个权值.3种操作. 1:修改某一个节点的权值. 2:询问某两个 ...
- [luogu2590][bzoj1036][ZJOI2008]树的统计
题目描述 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u ...
- 【lct】bzoj1036 [ZJOI2008]树的统计Count
题意:给你一棵树,点带权,支持三种操作:单点修改:询问链上和:询问链上max. 这里的Query操作用了与上一题不太一样的做法(上一题用那种做法,因为在边带权的情况下换根太困难啦): 先ChangeR ...
- bzoj1036 zjoi2008 树的统计 count
填坑= =第一道裸树剖 #include<cstdio> #include<algorithm> #include<cstring> #include<cst ...
随机推荐
- Python Map 并行
Map是一个酷酷的小东西,也是在Python代码轻松引入并行的关键.对此不熟悉的人会认为map是从函数式语言(如Lisp)借鉴来的东西.map是一个函数 - 将另一个函数映射到一个序列上.例如: ur ...
- 「日常训练」Watering Flowers(Codeforces Round #340 Div.2 C)
题意与分析 (CodeForces 617C) 题意是这样的:一个花圃中有若干花和两个喷泉,你可以调节水的压力使得两个喷泉各自分别以\(r_1\)和\(r_2\)为最远距离向外喷水.你需要调整\(r_ ...
- VIN码识别(车架号识别)在二手车交易中的应用
最新数据统计,2015年,中国卖出2110万辆新车,相比之下,美国卖出去了1740辆新车.然而,如果算上二手车,美国的汽车市场销量将扩展到4000多辆,而中国的乘用车才不到3000万辆. 销售总额上, ...
- centos7下搭建django
安装环境:centos7.4 1 安装nginx yum install nginx 注:尝试过在本地和腾讯云上安装,使用同一条命令:在本地安装提示没有可用安装包,云上安装正常 启动nginx,并启用 ...
- 【form】 表单组件说明
form表单组件 1)将form组件内的用户输入的<switch/> <input/> <checkbox/> <slider/> <radio/ ...
- 【WXS数据类型】Array
属性: 名称 值类型 说明 [Array].constructor [String] 返回值为“Array”,表示类型的结构字符串 [Array].length [Number] 返回数组长度 方法: ...
- Struts2(八.添加用户多张照片实现文件上传功能)
1.modify.jsp 在modify.jsp修改用户信息页面实现文件上传,添加用户照片的功能 如果是文件上传,method必须是post,必须指定enctype <form method=& ...
- 关于javascript的一个小问题,请问有人看出啥问题吗?
最近学习javascript,有一个问题挺奇怪的,先贴出代码: function binarySearch(){ var arr = [0,1,2,3]; var res = actbinarySea ...
- CsvHelper文档-1前言
CsvHelper文档-1前言 英文文档链接地址:CsvHelper Document 开源项目地址:CsvHelper 翻译于2018-1-5,原本可能会随时更新: 每一段代码都是经过我实际测试的, ...
- 今日头条 2018 AI Camp 6 月 2 日在线笔试编程题第一道——最大连续区间和扩展
题目 给出一个长度为 n 的数组a1.a2.....ana1.a2.....an,请找出在所有连续区间 中,区间和最大同时这个区间 0 的个数小于等于 3 个,输出这个区间和. 输入描述: 第一行一个 ...