HYSBZ 1036树链剖分
一棵树上有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
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树链剖分的更多相关文章
- HDU 3966 & POJ 3237 & HYSBZ 2243 树链剖分
树链剖分是一个很固定的套路 一般用来解决树上两点之间的路径更改与查询 思想是将一棵树分成不想交的几条链 并且由于dfs的顺序性 给每条链上的点或边标的号必定是连着的 那么每两个点之间的路径都可以拆成几 ...
- BZOJ 1036 && 树链剖分
还是太弱啊..各种数据结构只听过名字却没有一点概念..树链剖分也在这个范畴..今天来进一步深化一下教育改革推进全民素质提高. 性质 忘了在哪里看到的一篇blog有一句话讲得非常好,树链剖分不是一种数据 ...
- HYSBZ - 2243 树链剖分 + 线段树 处理树上颜色段数
用线段树处理颜色段数 记录区间内的颜色段数,区间右端点的颜色,区间右端点的颜色. int tr[maxn<<2], lc[maxn<<2], rc[maxn<<2] ...
- BZOJ 1036 [ZJOI2008]树的统计Count (树链剖分 - 点权剖分 - 单点权修改)
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1036 树链剖分模版题,打的时候注意点就行.做这题的时候,真的傻了,单词拼错检查了一个多小时 ...
- Luogu 2590 [ZJOI2008]树的统计 / HYSBZ 1036 [ZJOI2008]树的统计Count (树链剖分,LCA,线段树)
Luogu 2590 [ZJOI2008]树的统计 / HYSBZ 1036 [ZJOI2008]树的统计Count (树链剖分,LCA,线段树) Description 一棵树上有n个节点,编号分别 ...
- HYSBZ 1036 【树链剖分】
思路: 裸裸的树链剖分.... 树链剖分就是把一棵树分成若干重链和轻链...然后保证形成的线段树上每条链是连续存储的.然后这样就能用线段树进行维护了. 但是每次一定要保证是在同一条链里边....思路就 ...
- HYSBZ 1036(树链剖分)
题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?cid=28982#problem/E 题意:给定一棵树及树上的点权,要求三种操作: 1) ...
- HYSBZ 1036 树的统计Count(树链剖分)题解
思路: 树链剖分,不知道说什么...我连模板都不会用 代码: #include<map> #include<ctime> #include<cmath> #incl ...
- BZOJ 1036: [ZJOI2008]树的统计Count [树链剖分]【学习笔记】
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 14302 Solved: 5779[Submit ...
随机推荐
- Eclipse下载安装并运行第一个Hello world(详细)
Eclipse下载安装并运行第一个Hello world(详细) 1.下载安装和配置JDK JDK详细的安装教程参考:https://www.cnblogs.com/mxxbc/p/11845150. ...
- [视频演示].NET Core开发的iNeuOS物联网平台,实现从设备&PLC、云平台、移动APP数据链路闭环
目 录 1. 概述... 1 2. 登陆信息... 2 3. 设备驱动... 3 4. 组态建模... 3 5. 手机APP. 5 6. ...
- 不止面试—jvm类加载面试题详解
面试题 带着问题学习是最高效的,本次我们将尝试回答以下问题: 什么是类的加载? 哪些情况会触发类的加载? 讲一下JVM加载一个类的过程 什么时候会为变量分配内存? JVM的类加载机制是什么? 双亲委派 ...
- Python 常用模块系列学习(3)--configparser module
configpaser 模块----用于生成和修改常见配置文档 1. config 对象的创建: import configparser #导入模块 config = configparser.Con ...
- asp.net以流导出Excel
废话不多说,直接上代码 这是点击导出的事件函数,因为我是从前端获取的Table的json数据,所以需要转换一下,大家直接用查询出来的DataTable即可 protected void bt_expo ...
- Spring框架学习总结(上)
目录 1.Spring的概述 2.Spring的入门(IOC) 3.Spring的工厂类 4.Spring的配置 5.Spring的属性注入 6.Spring的分模块开发的配置 @ 1.Spring的 ...
- KDevelop
ctags+vim还是太累了,还是使用IDE好,尤其是c++模板.KDevelop就不错,符号智能推导以及cmake项目管理和配置,还是挺好用的. Android端的ndk开发使用Android St ...
- ubuntu server 1604 设置笔记本盒盖 不操作
sudo vim /etc/systemd/logind.conf //打开配置文件 找到 #HandleLidSwitch=suspend 改为 HandleLidSwitch=ignore ...
- 2019-11-7:练习上传getshell,通过菜刀连接
一.使用phtstudy搭建环境 1,将emlog源码解压到phpstudy的www目录,复制emlog源码中的src所有文件到需要搭建emlog的目录中,修改emlog配置文件,输入好本机mysql ...
- Selenium+Java(六)Selenium 强制等待、显式等待、隐实等待
前言 在实际测试过程中,由于网速或性能方面的原因,打开相应的网页后或在网页上做了相应的操作,网页上的元素可能不会马上加载出来,这个时候需要在定位元素前等待一下,等元素加载出来后再进行定位,根据实际使用 ...