一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w。我们将以下面的形式来要求你对这棵树完成
一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 I
II. QSUM u v: 询问从点u到点v的路径上的节点的权值和 注意:从点u到点v的路径上的节点包括u和v本身

Input

  输入的第一行为一个整数n,表示节点的个数。接下来n – 1行,每行2个整数a和b,表示节点a和节点b之间有
一条边相连。接下来n行,每行一个整数,第i行的整数wi表示节点i的权值。接下来1行,为一个整数q,表示操作
的总数。接下来q行,每行一个操作,以“CHANGE u t”或者“QMAX u v”或者“QSUM u v”的形式给出。 
对于100%的数据,保证1<=n<=30000,0<=q<=200000;中途操作中保证每个节点的权值w在-30000到30000之间。

Output

  对于每个“QMAX”或者“QSUM”的操作,每行输出一个整数表示要求输出的结果。

Sample Input

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

Sample Output

4

1

2

2

10

6

5

6

5

16

Hint

树链剖分+线段树模板题:

参考代码:

 //树链剖分+线段树
/*
I. CHANGE u t : 把结点u的权值改为t
II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 I
III. QSUM u v: 询问从点u到点v的路径上的节点的权值和 (包括u和v本身)
*/
#include<bits/stdc++.h>
using namespace std;
#define mem(a,b) memset(a,b,sizeof a)
#define mp make_pair
#define eps 1e-8
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
typedef long long ll;
typedef unsigned long long ull;
const int INF=0x3f3f3f3f;
const ll inf=0x3f3f3f3f3f3f3f3fll;
const int maxn=2e5+;
int siz[maxn],top[maxn],fa[maxn],dep[maxn];
int tid[maxn],rnk[maxn],son[maxn],cnt;
int head[maxn],tot,a[maxn]; struct Node{
int v,nxt;
} edge[maxn]; struct Tree{
int l,r,sz;
int sum,max_num;
} tree[maxn<<]; void Init()
{
tot=cnt=;
mem(head,-); mem(son,-);
mem(siz,); mem(top,);
} void Pushup(int pos)
{
tree[pos].max_num=max(tree[pos<<].max_num,tree[pos<<|].max_num);
tree[pos].sum=tree[pos<<].sum+tree[pos<<|].sum;
} void Build(int l,int r,int rt)
{
tree[rt].l=l;tree[rt].r=r;tree[rt].sz=r-l+;
if(l==r)
{
tree[rt].max_num=tree[rt].sum=a[rnk[l]];
return ;
}
int mid=(l+r)>>;
Build(lson); Build(rson);
Pushup(rt);
} void Update(int pos,int c,int l,int r,int rt)
{
if(l==r)
{
tree[rt].max_num=tree[rt].sum=c;
return ;
}
int mid=(l+r)>>;
if(pos<=mid) Update(pos,c,lson);
else Update(pos,c,rson);
Pushup(rt);
} int Query_sum(int L,int R,int l,int r,int rt)//求L~R的和
{
if(l>=L && r<=R) return tree[rt].sum;
int mid=(l+r)>>;
int ans=;
if(L<=mid) ans+=Query_sum(L,R,lson);
if(R>mid) ans+=Query_sum(L,R,rson);
return ans;
} int Query_max(int L,int R,int l,int r,int rt)//寻找L~R的最大值
{
if(L<=l && r<=R) return tree[rt].max_num;
int mid=(l+r)>>;
int ans=-INF;
if(L<=mid) ans=max(ans,Query_max(L,R,lson));
if(R>mid) ans=max(ans,Query_max(L,R,rson));
return ans;
} void addedge(int u,int v)
{
edge[tot].v=v;
edge[tot].nxt=head[u];
head[u]=tot++;
} void dfs1(int u,int father,int depth)
{
fa[u]=father;
dep[u]=depth;
siz[u]=;
for(int i=head[u];~i;i=edge[i].nxt)
{
int v=edge[i].v;
if(v!=fa[u])
{
dfs1(v,u,depth+);
siz[u]+=siz[v];
if(son[u]==- || siz[v]>siz[son[u]]) son[u]=v;
}
}
} void dfs2(int u,int t)
{
top[u]=t;
tid[u]=++cnt; rnk[cnt]=u;//tid[]:表示该点在线段树中的位置/:rnk[]:和tid[]数组相反,根据线段树中的位置,找到树中位置
if(son[u]==-) return ;
dfs2(son[u],t);
for(int i=head[u];~i;i=edge[i].nxt)
{
int v=edge[i].v;
if(v!=son[u] && v!=fa[u]) dfs2(v,v);
}
} int LCA(int u,int v,int flag)//沿着树链求LCA(根据flag的不同,求不同的值)
{
int fu=top[u],fv=top[v],res;
if(flag) res=-INF;// flag=1 求最大值
else res=; //flag=0 求和
while(fu!=fv)
{
if(dep[fu]<dep[fv]) swap(fu,fv),swap(u,v);
if(flag) res=max(res,Query_max(tid[fu],tid[u],,cnt,));
else res+=Query_sum(tid[fu],tid[u],,cnt,);
u=fa[fu],fu=top[u];
}
if(dep[u]>dep[v]) swap(u,v);
if(flag) res=max(res,Query_max(tid[u],tid[v],,cnt,));
else res+=Query_sum(tid[u],tid[v],,cnt,);
return res;
} int main()
{
int n,t,u,v,w;
char op[];
while(~scanf("%d",&n))
{
Init();
for(int i=;i<n;i++)
{
scanf("%d%d",&u,&v);
addedge(u,v);
addedge(v,u);
}
for(int i=;i<=n;i++) scanf("%d",&a[i]);
dfs1(,,);
dfs2(,);
Build(,cnt,);
scanf("%d",&t);
while(t--)
{
scanf("%s",op);
scanf("%d%d",&u,&v);
if(op[]=='S') printf("%d\n",LCA(u,v,));
else if(op[]=='M') printf("%d\n",LCA(u,v,));
else Update(tid[u],v,,cnt,);
}
}
return ;
}
  

HYSBZ 1036树链剖分的更多相关文章

  1. HDU 3966 & POJ 3237 & HYSBZ 2243 树链剖分

    树链剖分是一个很固定的套路 一般用来解决树上两点之间的路径更改与查询 思想是将一棵树分成不想交的几条链 并且由于dfs的顺序性 给每条链上的点或边标的号必定是连着的 那么每两个点之间的路径都可以拆成几 ...

  2. BZOJ 1036 && 树链剖分

    还是太弱啊..各种数据结构只听过名字却没有一点概念..树链剖分也在这个范畴..今天来进一步深化一下教育改革推进全民素质提高. 性质 忘了在哪里看到的一篇blog有一句话讲得非常好,树链剖分不是一种数据 ...

  3. HYSBZ - 2243 树链剖分 + 线段树 处理树上颜色段数

    用线段树处理颜色段数 记录区间内的颜色段数,区间右端点的颜色,区间右端点的颜色. int tr[maxn<<2], lc[maxn<<2], rc[maxn<<2] ...

  4. BZOJ 1036 [ZJOI2008]树的统计Count (树链剖分 - 点权剖分 - 单点权修改)

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1036 树链剖分模版题,打的时候注意点就行.做这题的时候,真的傻了,单词拼错检查了一个多小时 ...

  5. Luogu 2590 [ZJOI2008]树的统计 / HYSBZ 1036 [ZJOI2008]树的统计Count (树链剖分,LCA,线段树)

    Luogu 2590 [ZJOI2008]树的统计 / HYSBZ 1036 [ZJOI2008]树的统计Count (树链剖分,LCA,线段树) Description 一棵树上有n个节点,编号分别 ...

  6. HYSBZ 1036 【树链剖分】

    思路: 裸裸的树链剖分.... 树链剖分就是把一棵树分成若干重链和轻链...然后保证形成的线段树上每条链是连续存储的.然后这样就能用线段树进行维护了. 但是每次一定要保证是在同一条链里边....思路就 ...

  7. HYSBZ 1036(树链剖分)

    题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?cid=28982#problem/E 题意:给定一棵树及树上的点权,要求三种操作: 1) ...

  8. HYSBZ 1036 树的统计Count(树链剖分)题解

    思路: 树链剖分,不知道说什么...我连模板都不会用 代码: #include<map> #include<ctime> #include<cmath> #incl ...

  9. BZOJ 1036: [ZJOI2008]树的统计Count [树链剖分]【学习笔记】

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 14302  Solved: 5779[Submit ...

随机推荐

  1. redis 底层数据结构

    简单动态字符串SDS 包含字符串长度,剩余可用长度,字符数组 用于Redis中所有的string存储 字典(map) 数组+链表形式,跟hashMap很像 链地址法解决hash冲突 rehash使用新 ...

  2. iOS:应用程序扩展开发之Today扩展(Today Extesnsion)

    一.简介 iOS应用程序扩展是苹果在iOS8推出的一个新特性,可以将自定义的功能和内容扩展到应用程序之外,在之后又经过不断地优化和更新,已经成为开发中不可或缺的功能之一.扩展也是一个Target项目, ...

  3. 力扣(LeetCode)找不同 个人题解

    给定两个字符串 s 和 t,它们只包含小写字母. 字符串 t 由字符串 s 随机重排,然后在随机位置添加一个字母. 请找出在 t 中被添加的字母. 示例: 输入: s = "abcd&quo ...

  4. hopper逆向的伪代码令人大跌眼镜

    网上介绍hopper有逆向伪代码的文章很多,并以为其是万能而且cool B的.但是并没有人去求证hopper的逆向伪代码参考系数(参考价值,大家做过开发都清楚明白,有些功能看起来很花很cool但不实用 ...

  5. [转发]CSR 量产 烧录 软件

    蓝牙量产软件主要是为了应对蓝牙设备在批量生产时的一些如固件下载,地址下载,名字修改,以及一些辅助测试和检验功能. 目前,CSR推出的蓝牙芯片按照存储介质以及可编程与否分为两大类:ROM版本和Flash ...

  6. ubuntu 16.04上源码编译libjpeg-turbo和使用教程 | compile and use libjpeg-turbo on ubuntu 16.04

    本文首发于个人博客https://kezunlin.me/post/9f626e7a/,欢迎阅读! compile and use libjpeg-turbo on ubuntu 16.04 Seri ...

  7. 读懂JWT的使用,你就会用PHP如何实现了

    要如何用php实现JWT认证,那我们首先就来认识一下什么是JWT.什么是JWTJWT(json web token)是为了在网络应用环境间传递声明而执行的一种基于JSON的开放标准.JWT的声明一般被 ...

  8. 2019-10-30:渗透测试,基础学习,mssql堆叠内联注入,mongodb基础语法

    使用xp_cmdshell需要堆叠注入,http://192.168.190.148/less-1.asp?id=1';EXEC sp_configure 'show advanced options ...

  9. 新闻实时分析系统Hive与HBase集成进行数据分析

    (一)Hive 概述 (二)Hive在Hadoop生态圈中的位置 (三)Hive 架构设计 (四)Hive 的优点及应用场景 (五)Hive 的下载和安装部署 1.Hive 下载 Apache版本的H ...

  10. vue学习笔记(八)组件校验&通信

    前言 在上一章博客的内容中vue学习笔记(七)组件我们初步的认识了组件,并学会了如何定义局部组件和全局组件,上一篇内容仅仅只是对组件一个简单的入门,并没有深入的了解组件当中的其它机制,本篇博客将会带大 ...