[luogu2590][bzoj1036][ZJOI2008]树的统计
题目描述
一棵树上有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本身。
感想
感觉省选中的树剖题目还是有点多的呀,而且全都是模板题,巨简单~~,QAQ。
题解
今天看到了LCT的做法,感觉不会。写一篇树链剖分的题解。
既然知道是树剖,那么就只要套模板。
第一个操作:实际操作就是单点修改\(idx[u]\)
第二个操作,和第三个操作:都是树上的路径查询,还是模板。
\(ps.\)要注意,这个数存在负数,虽然对求和没有影响,但是在求最大值的时候需要从\(-inf\)开始。
ac代码
#include<bits/stdc++.h>
#define N 30005
#define LL long long
#define inf 0x3f3f3f3f
using namespace std;
int read(){
int w=0,x=0;char ch=0;
while(!isdigit(ch))w|=ch=='-',ch=getchar();
while(isdigit(ch))x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return w?-x:x;
}
int H[N],son[N],dep[N],top[N],sz[N],fa[N],idx[N],a[N],w[N];
int cnt,tot,n;
struct edge{
int to,nt;
}E[N<<1];
void addedge(int u,int v){
E[++cnt]=(edge){v,H[u]}; H[u]=cnt;
E[++cnt]=(edge){u,H[v]}; H[v]=cnt;
}
struct Segment_tree{
#define ls (nod<<1)
#define rs (nod<<1|1)
#define mid ((l+r)>>1)
struct node{
int l,r;
LL mx,s;
}tr[N<<2];
void pushup(int nod){tr[nod].s=tr[ls].s+tr[rs].s; tr[nod].mx=max(tr[ls].mx,tr[rs].mx);}
void build(int l,int r,int nod,int *a){
tr[nod].l=l; tr[nod].r=r;
if(l==r){tr[nod].mx=tr[nod].s=a[l]; return;}
build(l,mid,ls,a); build(mid+1,r,rs,a);
pushup(nod);
}
void update_point(int nod,int k,int w){
int l=tr[nod].l,r=tr[nod].r;
if(l==r){tr[nod].mx=tr[nod].s=w;return;}
if(k<=mid) update_point(ls,k,w);
else update_point(rs,k,w);
pushup(nod);
}
LL query_sec_sum(int nod,int ql,int qr){
int l=tr[nod].l,r=tr[nod].r; LL res=0;
if(ql<=l&&r<=qr) return tr[nod].s;
if(ql<=mid) res+=query_sec_sum(ls,ql,qr);
if(qr>mid) res+=query_sec_sum(rs,ql,qr);
return res;
}
LL query_sec_max(int nod,int ql,int qr){
int l=tr[nod].l,r=tr[nod].r; LL res=-inf;
if(ql<=l&&r<=qr) return tr[nod].mx;
if(ql<=mid) res=max(res,query_sec_max(ls,ql,qr));
if(qr>mid) res=max(res,query_sec_max(rs,ql,qr));
return res;
}
}tr;
void dfs1(int u,int ft,int dp){
fa[u]=ft; dep[u]=dp; sz[u]=1;
int maxson=-1;
for(int e=H[u];e;e=E[e].nt){
int v=E[e].to; if(v==fa[u]) continue;
dfs1(v,u,dp+1); sz[u]+=sz[v];
if(sz[v]>maxson) maxson=sz[v],son[u]=v;
}
}
void dfs2(int u,int tp){
top[u]=tp; idx[u]=++tot; a[tot]=w[u];
if(!son[u]) return; dfs2(son[u],tp);
for(int e=H[u];e;e=E[e].nt){
int v=E[e].to; if(v==fa[u]||v==son[u]) continue;
dfs2(v,v);
}
}
LL query_sum(int u,int v){
LL res=0;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]) swap(u,v);
res+=tr.query_sec_sum(1,idx[top[u]],idx[u]);
u=fa[top[u]];
}
if(dep[u]>dep[v]) swap(u,v);
res+=tr.query_sec_sum(1,idx[u],idx[v]);
return res;
}
LL query_max(int u,int v){
LL res=-inf;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]) swap(u,v);
res=max(res,tr.query_sec_max(1,idx[top[u]],idx[u]));
u=fa[top[u]];
}
if(dep[u]>dep[v]) swap(u,v);
res=max(res,tr.query_sec_max(1,idx[u],idx[v]));
return res;
}
int main(){
cnt=tot=0;
n=read();
for(int i=1;i<n;i++) addedge(read(),read());
for(int i=1;i<=n;i++) w[i]=read();
dfs1(1,0,1); dfs2(1,1); tr.build(1,n,1,a);
int q=read();
while(q--){
char opt[10]; scanf("%s",opt); int u=read(),x=read();
if(opt[0]=='C') tr.update_point(1,idx[u],x);
else{
if(opt[1]=='M') printf("%lld\n",query_max(u,x));
else printf("%lld\n",query_sum(u,x));
}
}
return 0;
}
[luogu2590][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 树链剖分
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1036 题意概括 一个树,每个节点有一个权值.3种操作. 1:修改某一个节点的权值. 2:询问某两个 ...
- 【lct】bzoj1036 [ZJOI2008]树的统计Count
题意:给你一棵树,点带权,支持三种操作:单点修改:询问链上和:询问链上max. 这里的Query操作用了与上一题不太一样的做法(上一题用那种做法,因为在边带权的情况下换根太困难啦): 先ChangeR ...
- BZOJ1036[ZJOI2008]树的统计Count 题解
题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.有一些操作:1.把结点u的权值改为t:2.询问从点u到点v的路径上的节点的最大权值 3.询问从点u到点v的路径上的节点的权值和 ...
- bzoj1036 zjoi2008 树的统计 count
填坑= =第一道裸树剖 #include<cstdio> #include<algorithm> #include<cstring> #include<cst ...
- bzoj1036 [ZJOI2008]树的统计
一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从 ...
随机推荐
- Pandas简易入门(一)
目录: 读取数据 索引 选择数据 简单运算 声明,本文引用于:https://www.dataquest.io/mission/8/introduction-to-pandas (建议阅读原文) Pa ...
- PHP生成QRCode二维码
php生成QRCode二维码示例 <?php //引入 phpqrcode 类库 //phpqrcode下载地址:https://github.com/t0k4rt/phpqrcode //或从 ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(63)-WebApi与Unity注入
系列目录 前言: 有时候我们系统需要开放数据给手机App端或其他移动设备,不得不说Asp.net WebApi是目前首选 本节记录Asp.net MVC WebApi怎么利用Unity注入.系列开头已 ...
- Openstack架构概念图-简单汇总
OpenStack是一个云平台管理的项目,它不是一个软件.这个项目由几个主要的组件组合起来完成一些具体的工作.想要了解openstack,第一步我们可以观察他的概念图: 针对上图的翻译+解释: 上图主 ...
- PAT甲级题解-1123. Is It a Complete AVL Tree (30)-AVL树+满二叉树
博主欢迎转载,但请给出本文链接,我尊重你,你尊重我,谢谢~http://www.cnblogs.com/chenxiwenruo/p/6806292.html特别不喜欢那些随便转载别人的原创文章又不给 ...
- Visual Studio2013的安装过程及练习测试
一.安装环境: 支持安装的操作系统版本:Windows XP,Windows7,Windows8,Windows10. CPU大小:Intel(R)Core(TM)i5-4210U CPU @1.7G ...
- Zookeeper 源码学习(一)环境搭建
前言 最近准备学习 Zookeeper,想从 Zookeeper 开始逐步深入了解各类中间件,学习分布式计算. 下载源码 执行指令,下载代码: git clone https://github.com ...
- linux第一次读书笔记
第一章 LINUX内核简介 1.1 Unix的历史 1969年的夏天,贝尔实验室的程序员们在一台PDR-7型机上实现了Unix这个全新的操作系统. 1973年,整个Unix系统用C语言进行了重写,给后 ...
- Linux内核分析 期末总结
Linux内核分析 期末总结 一.知识概要 1. 计算机是如何工作的 存储程序计算机工作模型:冯诺依曼体系结构 X86汇编基础 会变一个简单的C程序分析其汇编指令执行过程 2. 操作系统是如何工作的 ...
- 3-palindrome CodeForces - 805B (思维)
In the beginning of the new year Keivan decided to reverse his name. He doesn't like palindromes, so ...