题目:

题目背景

ZJOI2008 DAY1 T4

题目描述

一棵树上有 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”的操作,每行输出一个整数表示要求输出的结果。

样例数据 1

输入  [复制]

 


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

输出





10 




16

备注

【数据范围】
对于 100% 的数据,保证1<=n<=30000;0<=q<=200000;中途操作中保证每个节点的权值 w 在 -30000 到 30000 之间。

方法:

树链剖分模板题;

代码:

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cmath>
#include<ctime>
#include<cctype>
#include<cstring>
#include<string>
#include<algorithm>
using namespace std;
const int N=3e4+;
const int inf=1e+;
int n,Q;
int first[N],next[N*],go[N*];
int size[N],top[N],father[N],pos[N],idx[N],deep[N],tot,son[N],val[N];
int summ[N*],maxx[N*];
char st[]; inline void combin(int u,int v)
{
next[++tot]=first[u],first[u]=tot,go[tot]=v;
next[++tot]=first[v],first[v]=tot,go[tot]=u;
} inline void dfs1(int u)
{
size[u]=;
for(int e=first[u],v;e;e=next[e])
{
if((v=go[e])==father[u]) continue;
father[v]=u;
deep[v]=deep[u]+;
dfs1(v);
size[u]+=size[v];
if(size[v]>size[son[u]]) son[u]=v;
}
} inline void dfs2(int u)
{
if(son[u])
{
idx[pos[son[u]]=++tot]=son[u];
top[son[u]]=top[u];
dfs2(son[u]);
}
for(int e=first[u],v;e;e=next[e])
{
if((v=go[e])==father[u]||v==son[u]) continue;
idx[pos[v]=++tot]=v;
top[v]=v;
dfs2(v);
}
} inline void pre()
{
dfs1();
top[]=pos[]=idx[]=tot=;
dfs2();
} inline void build(int k,int l,int r)
{
if(l==r)
{
summ[k]=maxx[k]=val[idx[l]];
return;
}
int mid=(l+r)/;
build(k*,l,mid);
build(k*+,mid+,r);
summ[k]=summ[k*]+summ[k*+];
maxx[k]=max(maxx[k*],maxx[k*+]);
} inline void modify(int k,int l,int r,int p,int v)
{
if(l==r)
{
summ[k]=maxx[k]=v;
return;
}
int mid=(l+r)/;
if(p<=mid) modify(k*,l,mid,p,v);
else modify(k*+,mid+,r,p,v);
summ[k]=summ[k*]+summ[k*+];
maxx[k]=max(maxx[k*],maxx[k*+]);
} inline int querymax(int k,int l,int r,int x,int y)
{
if(l>=x&&r<=y)
return maxx[k];
int mid=(l+r)/,res=-inf;
if(x<=mid) res=querymax(k*,l,mid,x,y);
if(y>mid) res=max(res,querymax(k*+,mid+,r,x,y));
return res;
} inline int querysum(int k,int l,int r,int x,int y)
{
if(l>=x&&r<=y)
return summ[k];
int mid=(l+r)/,res=;
if(x<=mid) res+=querysum(k*,l,mid,x,y);
if(y>mid) res+=querysum(k*+,mid+,r,x,y);
return res;
} inline int pathmax(int u,int v)
{
if(top[u]!=top[v])
{
if(deep[top[u]]<deep[top[v]]) swap(u,v);
return max(pathmax(father[top[u]],v),querymax(,,n,pos[top[u]],pos[u]));
}
if(deep[u]>deep[v]) swap(u,v);
return querymax(,,n,pos[u],pos[v]);
} inline int pathsum(int u,int v)
{
if(top[u]!=top[v])
{
if(deep[top[u]]<deep[top[v]]) swap(u,v);
return pathsum(father[top[u]],v)+querysum(,,n,pos[top[u]],pos[u]);
}
if(deep[u]>deep[v]) swap(u,v);
return querysum(,,n,pos[u],pos[v]);
} int main()
{
//freopen("a.in","r",stdin);
//freopen("a.out","w",stdout);
scanf("%d",&n);
int u,v;
for(int i=;i<n;i++)
{
scanf("%d%d",&u,&v);
combin(u,v);
}
for(int i=;i<=n;i++)
scanf("%d",&val[i]);
pre();
build(,,n);
scanf("%d",&Q);
while(Q--)
{
scanf("%s%d%d",st,&u,&v);
if(st[]=='M')
printf("%d\n",pathmax(u,v));
if(st[]=='S')
printf("%d\n",pathsum(u,v));
if(st[]=='H')
modify(,,n,pos[u],v);
}
return ;
}

算法复习——树链剖分模板(bzoj1036)的更多相关文章

  1. BZOJ 2243 染色 | 树链剖分模板题进阶版

    BZOJ 2243 染色 | 树链剖分模板题进阶版 这道题呢~就是个带区间修改的树链剖分~ 如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了. 这道题要注意的是,无论是线段树上 ...

  2. Hdu 5274 Dylans loves tree (树链剖分模板)

    Hdu 5274 Dylans loves tree (树链剖分模板) 题目传送门 #include <queue> #include <cmath> #include < ...

  3. bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题

    [ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...

  4. bzoj1036 树的统计 树链剖分模板

    题意:给出树上任意两点,求路径上的值的和与最大值,带单点修改操作 树链剖分思路: 1.对树进行dfs求出点的深度和父亲节点,然后求出轻重儿子(重儿子就是点最多的那个子树,其余都是轻儿子),用一个son ...

  5. BZOJ 1036 [ZJOI2008]树的统计Count | 树链剖分模板

    原题链接 树链剖分的模板题:在点带权树树上维护路径和,最大值和单点修改 这里给出几个定义 以任意点为根,然后记 size (u ) 为以 u 为根的子树的结点个数,令 v 为 u 所有儿子中 size ...

  6. BZOJ 1036 树的统计Count 树链剖分模板题

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1036 题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将 ...

  7. 树链剖分学习&BZOJ1036

    题目传送门 树链剖分,计算机术语,指一种对树进行划分的算法,它先通过轻重边剖分将树分为多条链,保证每个点属于且只属于一条链,然后再通过数据结构(树状数组.SBT.SPLAY.线段树等)来维护每一条链. ...

  8. spoj - Grass Planting(树链剖分模板题)

    Grass Planting 题意 给出一棵树,树有边权.每次给出节点 (u, v) ,有两种操作:1. 把 u 到 v 路径上所有边的权值加 1.2. 查询 u 到 v 的权值之和. 分析 如果这些 ...

  9. SPOJ QTREE - Query on a tree 【树链剖分模板】

    题目链接 引用到的大佬博客 代码来自:http://blog.csdn.net/jinglinxiao/article/details/72940746 具体算法讲解来自:http://blog.si ...

随机推荐

  1. java代码(处理json串)

    package test; import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.JSONObject; public class ...

  2. Mysql的Root密码忘记,查看或修改的解决方法

    Mysql的Root密码忘记,查看或修改的解决方法:1.首先启动命令行2.在命令行运行:taskkill /f /im mysqld-nt.exe3.继续在命令行运行:mysqld-nt --skip ...

  3. com.alibaba.dubbo.remoting.RemotingException: Failed to bind NettyServer on /192.168.1.13:20881, cause: Failed to bind to: /0.0.0.0:20881

    抛出的异常如上,解决方案是:根据异常信息确定是端口被占用,排查项目是否启动之后没有关闭,在windows命令行中运行如下命令:netstat -ano 检查端口占用的情况,根据pid在任务管理器中杀死 ...

  4. poj1338 Ugly Numbers 打表, 递推

    题意:一个数的质因子能是2, 3, 5, 那么这个数是丑数. 思路: 打表或者递推. 打表: 若该数为丑数,那么一定能被2 或者3, 或者5 整除, 除完之后则为1. #include <ios ...

  5. windows/Linux 常用命令

    windows 文件操作命令 cd 切换文件目录 dir 显示文件目录内容 md 创建文件夹 rd 删除文件夹 copy 拷贝文件 move 移动文件 del 删除文件 replace 替换文件 mk ...

  6. 栈的应用——Rails

    一.题目描述 某城市有一个火车站,有n节车厢从A方向驶入车站,按进站顺序编号为1~n,经中转站C驶向B.中转站C,这是一个可以停放任意多节车厢的车站,但由于末端封顶,驶入C的车厢必须以相反的顺序驶出C ...

  7. python Object-Oriented Programming

    Python 类的成员.成员修饰符.类的特殊成员. Python 类的成员 类的成员可以分为三大类: 字段.方法和属性. #注:所有成员中,只有普通字段的内容保存对象中,即: #根据此类创建了多少对象 ...

  8. bootstrap历练实例:按钮作为输入框组前缀或后缀

    <!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...

  9. Python使用三种方法实现PCA算法[转]

    主成分分析(PCA) vs 多元判别式分析(MDA) PCA和MDA都是线性变换的方法,二者关系密切.在PCA中,我们寻找数据集中最大化方差的成分,在MDA中,我们对类间最大散布的方向更感兴趣. 一句 ...

  10. [CODEVS] 3955 最长严格上升子序列(加强版)

    题目描述 Description 给一个数组a1, a2 ... an,找到最长的上升降子序列ab1<ab2< .. <abk,其中b1<b2<..bk. 输出长度即可. ...