[luogu P2590 ZJOI2008] 树的统计 (树链剖分)

题目描述
一棵树上有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:
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
输出样例#1:
4
1
2
2
10
6
5
6
5
16
说明
对于100%的数据,保证1<=n<=30000,0<=q<=200000;中途操作中保证每个节点的权值w在-30000到30000之间。

树链剖分模板。。。
code:

//By Menteur_Hxy
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<vector>
#include<cmath>
#define ll long long
#define f(a,b,c) for(int a=b;a<=c;a++)
using namespace std;inline ll rd() {ll x=0,fla=1; char c=' ';while(c>'9' || c<'0') {if(c=='-') fla=-fla; c=getchar();}while(c<='9' && c>='0') x=x*10+c-'0',c=getchar();return x*fla;
}inline void out(ll x){int a[25],wei=0;if(x<0) putchar('-'),x=-x;for(;x;x/=10) a[++wei]=x%10;if(wei==0){ puts("0"); return;}for(int j=wei;j>=1;--j) putchar('0'+a[j]);putchar('\n');
}const int MAX=300100;//直接开了十倍qwq
const int INF=0x3f3f3f3f;
int n,cnt,N;int head[MAX],W[MAX],size[MAX],h[MAX],fa[MAX],son[MAX];
//W 点权 size 树的大小 h 深度 fa 父亲 son 重儿子 int num[MAX],top[MAX],tree[MAX],maxn[MAX],sumn[MAX];
//num 在线段树编号 top 链最上面的点 tree 线段树编号对应的点 struct edges{int next,to;
}edge[MAX]; void add(int a,int b) {edge[++cnt]=(edges) {head[a],b}; head[a]=cnt;edge[++cnt]=(edges) {head[b],a}; head[b]=cnt;
}void dfs1(int u,int pre,int dep) {size[u]=1; h[u]=dep; fa[u]=pre;//initint mason=0;for(int i=head[u];i;i=edge[i].next) if(edge[i].to!=pre) {int v=edge[i].to;dfs1(v,u,dep+1);size[u]+=size[v];if(size[v]>mason) {mason=size[v];son[u]=v;}}
}void dfs2(int u,int pre) {if(son[pre]!=u) top[u]=u;else top[u]=top[pre];num[u]=++N;if(son[u]) dfs2(son[u],u);for(int i=head[u];i;i=edge[i].next)if(edge[i].to!=pre && edge[i].to !=son[u])dfs2(edge[i].to,u);
}void build_sum(int cur,int l,int r) {if(l==r) {sumn[cur]=W[tree[l]];return ;}int mid=(l+r)>>1;build_sum(cur<<1,l,mid);build_sum(cur<<1|1,mid+1,r);sumn[cur]=sumn[cur<<1]+sumn[cur<<1|1];//update_sum
}void build_max(int cur,int l,int r) {if(l==r) {maxn[cur]=W[tree[l]];return ;}int mid=(l+r)>>1;build_max(cur<<1,l,mid);build_max(cur<<1|1,mid+1,r);maxn[cur]=max(maxn[cur<<1],maxn[cur<<1|1]);//update_max
}void po_ch_sum(int cur,int l,int r,int x,int v) {//point_change_sumif(l==r) {sumn[cur]=v;return ;}int mid=(l+r)>>1;if(x<=mid) po_ch_sum(cur<<1,l,mid,x,v);else po_ch_sum(cur<<1|1,mid+1,r,x,v);sumn[cur]=sumn[cur<<1]+sumn[cur<<1|1];//update_sum
}void po_ch_max(int cur,int l,int r,int x,int v) {//point_change_maxif(l==r) {maxn[cur]=v;return ;}int mid=(l+r)>>1;if(x<=mid) po_ch_max(cur<<1,l,mid,x,v);else po_ch_max(cur<<1|1,mid+1,r,x,v);maxn[cur]=max(maxn[cur<<1],maxn[cur<<1|1]);
}int query_sum(int cur,int l,int r,int L,int R) {if(L<=l&&r<=R) return sumn[cur];int mid=(l+r)>>1,ans=0;if(L<=mid) ans+=query_sum(cur<<1,l,mid,L,R);if(R>mid) ans+=query_sum(cur<<1|1,mid+1,r,L,R);return ans;
}int query_max(int cur,int l,int r,int L,int R) {if(L<=l&&r<=R) return maxn[cur];int mid=(l+r)>>1,ans=-INF;if(L<=mid) ans=max(ans,query_max(cur<<1,l,mid,L,R));if(R>mid) ans=max(ans,query_max(cur<<1|1,mid+1,r,L,R));return ans;
}void INIT() {dfs1(1,0,1);// size h fa sondfs2(1,0);//top numf(i,1,n) tree[num[i]]=i;//tree//建树: build_sum(1,1,N);build_max(1,1,N);
}void solve() {int q=rd(),a,b,ans=0,f1,f2;char opt[6];f(i,1,q) {scanf("%s",opt);a=rd(),b=rd(),ans=0;if(opt[0]=='C') {//HCANGEpo_ch_sum(1,1,N,num[a],b);po_ch_max(1,1,N,num[a],b);}else {f1=top[a],f2=top[b];if(opt[1]=='M') ans=-INF;while(f1!=f2) {if(h[f1]<h[f2]) {swap(a,b);swap(f1,f2);}if(opt[1]=='S') ans+=query_sum(1,1,N,num[f1],num[a]);else ans=max(ans,query_max(1,1,N,num[f1],num[a]));a=fa[f1];f1=top[a];}if(num[a]>num[b]) swap(a,b);if(opt[1]=='S') ans+=query_sum(1,1,N,num[a],num[b]);else ans=max(ans,query_max(1,1,N,num[a],num[b]));out(ans);}}
}int main() {n=rd();f(i,1,n-1) add(rd(),rd());f(i,1,n) W[i]=rd();INIT();solve();return 0;
}

转载于:https://www.cnblogs.com/Menteur-Hxy/p/9247981.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/452838.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

jetty xml解析

1 configure configure为xml的根结点&#xff0c;class指定所配置的对象的类&#xff0c;这个configure会创建一个该类的对象&#xff0c;然后根据该xml对其进行配置。id用来对该对象进行标识&#xff0c;在整个jetty中具有唯一性&#xff0c;相同id的xml configure文件配置的是…

java 歌词_请问吧内有大神用JAVA做过桌面歌词吗

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼写了个简单的例子给你&#xff1a;public class TextChangePane extends JComponent implements ActionListener {private static final int CYCLE_TIME 10000;private long startTime 0;private long nowTime 0;private float …

组播相对于单播和广播的优势

组播协议允许将一台主机发送的数据通过网络路由器和交换机复制到多个加入此组组播协议。 与现在广泛使用的单播协议的不同之处在于&#xff0c;一个主机用单播协议向n个主机发送相同的数据时&#xff0c;发送主机需要分别向n个主机发送&#xff0c;共发送n次。一个主机用组播协…

安装nginx及fastdfs-nginx-module

1.先了解背景&#xff1a; FastDFS为什么要结合Nginx以及FastDFS原理&#xff0c;请参考文章&#xff1a; FastDFS为什么要结合Nginx以及FastDFS原理 2.准备工作&#xff1a; 安装安装Nginx所需的环境&#xff0c;参考文献&#xff1a;Ubuntu 18.04.1安装Nginx apt install …

如何让自己的内心强大起来

内心强大的人是指一个人的精神境界达到了一定的级别&#xff01;以至于让人们折服&#xff01; 世界上有这么一种人&#xff0c;似乎特别得到老天爷的偏爱——他总是有自己的理想&#xff0c;并且总是努力去做&#xff0c;最重要的是&#xff0c;老天爷每一次都会帮他取得成功…

什么是软件工程

软件工程是指导计算机软件开发和维护的一门工程学科。采用工程的概念、原理、技术和方法来开发与维护软件&#xff0c;把经过时间考验而证明正确的管理技术和当前能够得到的最好的技术方法结合起来&#xff0c;以经济地开发出高质量的软件并有效地维护它&#xff0c;这就是软件…

linux下的静态库与动态库

目录 静态库定义&#xff1a;生成及使用方法&#xff1a;静态库的优缺点动态库定义&#xff1a;生成及使用方法&#xff1a;动态库优缺点&#xff1a;静态库 先说说我们为什么需要库&#xff1f; 当有些代码我们大量会在程序中使用比如&#xff08;scanf&#xff0c;printf等&a…

esrgan_ESRGAN: Enhanced Super-Resolution Generative Adversarial Networks【阅读笔记】

针对SRGAN提出的几点改进&#xff0c;获得了PIRM2018视觉质量的第一名。首先是使用去掉BN层的Residual in Residual Dense Block作为网络的basic unit。并且使用residual scling 和 smaller initialization帮助训练更深的网络。第二点改进是使用了Relativistic Discriminator来…

PostgreSQL Frontend/Backend protocol (通信协议)

标签 PostgreSQL , protocol , proxy , 通信协议 背景 理解PostgreSQL的通信协议可以更好的开发类似SQL代理&#xff0c;SQL中间件&#xff0c;SQL防火墙&#xff0c;连接池等软件。 学习资料与软件 《PostgreSQL 读写分离代理 - Crunchy Proxy(base on golang)》 Postgres on …

启动FastDFS服务,使用python客户端对接fastdfs完成上传测试

1.启动tracker、storage、nginx服务&#xff1a; 启动fdfs_trackerd&#xff1a;sudo service fdfs_trackerd start 启动fdfs_storaged &#xff1a;sudo service fdfs_storaged start 启动Nginx&#xff1a;sudo /usr/local/nginx/sbin/nginx 注&#xff1a;此处给出重启服务…

软件工程方法学

传统方法学 传统方法学也称为生命周期方法学或结构化范型。它采用结构化技术(结构化分析、结构化设计和结构化实现)来完成软件开发的各项任务&#xff0c;并使用适当的软件工具或软件工程环境来支持结构化技术的运用。 面向对象方法学 与传统方法相反&#xff0c;面向对象方…

我做项目这些年的经验

1、中国充满大量非常敬业但不够职业的项目经理&#xff0c;不了解这一点&#xff0c;就做不好中国的项目。 2、真正的原因往往都隐藏在表面的理由背后。 3、做项目最高境界是和用户形成长期共生双赢关系。 4、卖功能&#xff0c;卖利益&#xff0c;卖服务&#xff0c;卖价值…

Python学习-终端字体高亮显示

1、采用原生转义字符序列&#xff0c;对Windows有的版本不支持&#xff08;比如win7&#xff09;&#xff0c;完美支持Linux 实现过程&#xff1a; 终端的字符颜色是用转义序列控制的&#xff0c;是文本模式下的系统显示功能&#xff0c;和具体的语言无关。 转义序列是以ESC开头…

Win32-Application的窗口和对话框

Win32 Application&#xff0c;没有基于MFC的类库&#xff0c;而是直接调用C接口来编程。 一、弹出消息窗口 &#xff08;1&#xff09;最简单的&#xff0c;在当前窗口中弹出新窗口。新窗口只有“YES”按钮。 int APIENTRY WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstan…

Python面试题总结(4)--数据类型(列表)

1. 已知 AList [1,2,3,1,2]&#xff0c;对 AList 列表元素去重&#xff0c;写出具体过程。 答&#xff1a; AList [1,2,3,1,2] BList set(AList)print(BList) print(list(BList))输出结果&#xff1a; {1, 2, 3} [1, 2, 3]2. 如何实现 “1,2,3” 变成 [“1”,“2”,“3”…

项目团队要以十当一

如何建立起一支高效的团队&#xff0c;并有效的管理团队&#xff0c;一直是IT项目经理津津乐道的话题。任何一个IT项目经理对此都有自己一番不同的见解&#xff0c;根据自己团队特点&#xff0c;项目经理正在用自身独有的管理艺术改变着自己的团队。项目团队要以十当一&#xf…

Centos中配置环境变量

以Java的开发环境Jdk为例。 将jdk-9.0.1放置在/usr/local下&#xff08;UNIX规范&#xff09;&#xff0c;然后我们将jdk配置到环境变量中去。 $ mv jdk-9.0.1 /usr/local $ vim /etc/profile 修改 /etc/profile &#xff0c;最底部加入以下内容 export JAVA_HOME/usr/local/jd…

python面试题总结(5)--数据类型(字典)

1. 字典操作中 del 和 pop 有什么区别 答&#xff1a;del 可以根据索引&#xff08;元素所在位置&#xff09;来删除的&#xff0c;没有返回值。 pop 可以根据索引弹出一个值&#xff0c;然后可以接收它的返回值。 参考一 参考二 2. 按照字典的内的年龄排序 d1 [ {‘name’…

js下载文件 java_[Java教程]使用js实现点击按钮下载文件

[Java教程]使用js实现点击按钮下载文件0 2016-11-11 19:02:54有时候我们在网页上需要增加一个下载按钮&#xff0c;让用户能够点击后下载页面上的资料&#xff0c;那么怎样才能实现功能呢&#xff1f;这里有两种方法&#xff1a;现在需要在页面上添加一个下载按钮&#xff0c;点…