[置顶] bzoj 1036 树的统计Count 点权值模板
树链剖分 点权型可做模板,链路剖分的思想把点hash到线段树的上,然后可通过n*(log(n)*log(n))的复杂度在树上操作,在线段树上能操作的在链路上都能操作。
#include<cstdio>
#include<cstring>
#include<iostream>
#define M 40000
#define lson (rt<<1)
#define rson (rt<<1|1)
#define inf 0x3f3f3f3f
using namespace std;
struct G{
int head[M],en;
struct E{
int u,v,next,cost;
}e[M<<2];
void init(){
memset(head,-1,sizeof(head));en=0;
}
void add(int u,int v,int cost){
e[en].u=u;e[en].v=v;e[en].cost=cost;e[en].next=head[u];head[u]=en++;
}
}g1;
struct A{
int tsum,tmax;
A(){
tsum=0;
tmax=-inf;
}
};
struct T{
struct E{
int l,r,tmax,tsum;
}e[M<<2];
int value[M];
void push_up(int rt){
e[rt].tmax=max(e[lson].tmax,e[rson].tmax);
e[rt].tsum=e[lson].tsum+e[rson].tsum;
}
void build(int l,int r,int rt){
e[rt].l=l;e[rt].r=r;e[rt].tmax=-inf;e[rt].tsum=0;
if(l==r){
e[rt].tmax=value[l];
e[rt].tsum=value[l];
return ;
}
int mid=(l+r)>>1;
build(l,mid,lson);
build(mid+1,r,rson);
push_up(rt);
}
void update(int ul,int new_flag,int rt){
if(e[rt].l==ul&&e[rt].r==ul){
e[rt].tmax=new_flag;
e[rt].tsum=new_flag;
return ;
}
int mid=(e[rt].l+e[rt].r)>>1;
if(ul<=mid) update(ul,new_flag,lson);
else update(ul,new_flag,rson);
push_up(rt);
}
A query(int ul,int ur,int rt){
if(ul<=e[rt].l&&ur>=e[rt].r){
A a1;a1.tmax=e[rt].tmax;
a1.tsum=e[rt].tsum;
return a1;
}
int mid=(e[rt].l+e[rt].r)>>1;
if(ur<=mid) return query(ul,ur,lson);
else if(ul>mid) return query(ul,ur,rson);
else{
A a1=query(ul,ur,lson);
A a2=query(ul,ur,rson);
a1.tmax=max(a1.tmax,a2.tmax);
a1.tsum+=a2.tsum;
return a1;
}
}
}tr;
int my_que[M<<2],top[M],son[M],en,sz[M],fa[M],dep[M],hash[M],value[M];
void build_tree(){
int l=0,r=-1;
my_que[++r]=1;
dep[1]=1;
sz[0]=0;
fa[1]=0;
while(l<=r){
int u=my_que[l++];son[u]=0;sz[u]=1;
for(int i=g1.head[u];i!=-1;i=g1.e[i].next){
int v=g1.e[i].v;
if(v==fa[u]) continue;
fa[v]=u;
dep[v]=dep[u]+1;
my_que[++r]=v;
}
}
for(int i=r;i>=0;i--){
int u=my_que[i];
for(int j=g1.head[u];j!=-1;j=g1.e[j].next){
int v=g1.e[j].v;
if(v==fa[u]) continue;
sz[u]+=sz[v];
if(sz[son[u]]<sz[v]) son[u]=v;
}
}
for(int i=0;i<=r;i++){
int u=my_que[i];
if(son[fa[u]]==u) top[u]=top[fa[u]];
else top[u]=u;
}
}
void re_build(int n){
en=1;
for(int i=1;i<=n;i++){
if(top[i]!=i) continue;
for(int j=i;j;j=son[j]){
hash[j]=en;
tr.value[en++]=value[j];
}
}
tr.build(1,en,1);
}
void work(int n){
build_tree();
re_build(n);
}
void change(int l,int new_flag){
tr.update(hash[l],new_flag,1);
}
A query(int x,int y){
A ans,a1;;
while(top[x]!=top[y]){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
a1=tr.query(hash[top[x]],hash[x],1);
ans.tmax=max(ans.tmax,a1.tmax);
ans.tsum+=a1.tsum;
x=fa[top[x]];
}
if(dep[y]<dep[x]) swap(x,y);
a1=tr.query(hash[x],hash[y],1);
ans.tmax=max(ans.tmax,a1.tmax);
ans.tsum+=a1.tsum;
return ans;
}
int main(){
int n;while(~scanf("%d",&n)){
g1.init();
for(int i=0;i<n-1;i++){
int u,v;scanf("%d%d",&u,&v);
g1.add(u,v,0);
g1.add(v,u,0);
}
for(int i=1;i<=n;i++) scanf("%d",&value[i]);
work(n);
int q;scanf("%d",&q);
for(int i=0;i<q;i++){
char str[100];
scanf("%s",str);
if(str[0]=='C'){
int t1,t2;scanf("%d%d",&t1,&t2);
change(t1,t2);
}
else{
int t1,t2;scanf("%d%d",&t1,&t2);
A ans;ans=query(t1,t2);
if(str[1]=='M'){
printf("%d\n",ans.tmax);
}
else{
printf("%d\n",ans.tsum);
}
}
}
}
return 0;
}
[置顶] bzoj 1036 树的统计Count 点权值模板的更多相关文章
- BZOJ 1036 树的统计Count 树链剖分模板题
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1036 题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将 ...
- bzoj 1036 树的统计Count
题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u ...
- BZOJ - 1036 树的统计Count (LCT)
LCT试炼题(代码量居然完爆树剖?) #include<bits/stdc++.h> using namespace std; ,inf=0x3f3f3f3f; ],flp[N],n,m, ...
- BZOJ - 1036 树的统计Count (树链剖分+线段树)
题目链接 #include<bits/stdc++.h> using namespace std; typedef long long ll; ,inf=0x3f3f3f3f; ],mx[ ...
- BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)
BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...
- BZOJ 1036 树的统计-树链剖分
[ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 12904 Solved: 5191[Submit][Status ...
- BZOJ 1036 树的统计
Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. Q ...
- Codevs 2460 == BZOJ 1036 树的统计
2460 树的统计 2008年省队选拔赛浙江 时间限制: 2 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 一棵树上有n个节点,编号分别为1 ...
- Bzoj 1036 树的统计 分类: ACM TYPE 2014-12-29 18:55 72人阅读 评论(0) 收藏
Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. Q ...
随机推荐
- IDA Pro 权威指南学习笔记(十) - 栈帧
栈帧(stack frame)是在程序的运行时栈中分配的内存块,用于特定的函数调用 如果一个函数没有执行则不需要内存,当函数被调用时就需要用到内存 1.传给函数的参数的值需要存储到函数能够找到它们的位 ...
- ceph集群jewel版本 rbd 块map 报错-故障排查
测试信息如下: [root@ceph_1 ~]# ceph osd pool lsrbdchy_123swimmingpool #新建rbd 块: rbd create swimmingpool/ba ...
- 11.solr学习速成之MoreLikeThis
Solr相似匹配 在网页搜索或电商产品搜索结果页面,很多时候会看到一个相似文档.相似产品或找相似的链接.Solr 使用 MoreLikeThisComponent(MLT)和 MoreLikeT ...
- 机器学习,数据挖掘,统计学,云计算,众包(crowdsourcing),人工智能,降维(Dimension reduction)
机器学习 Machine Learning:提供数据分析的能力,机器学习是大数据时代必不可少的核心技术,道理很简单:收集.存储.传输.管理大数据的目的,是为了“利用”大数据,而如果没有机器学习技术分析 ...
- Django的contenttypes应用、缓存相关
一.django的contenttypes contenttypes 是Django内置的一个应用 , 可以追踪项目中所有app 和 model 的对应关系, 并记录djang_content_typ ...
- 一次性show 出所有配置
cisco的全页打印显示配置信息的命令: #terminal length 0 #show run 华为和H3C的全页打印显示配置信息的命令: ]user-interface vty 0 4 ]scr ...
- 教你用一行Python代码实现并行(转)
教你用一行Python代码实现并行 本文教你通过一行Python实现并行化. Python在程序并行化方面多少有些声名狼藉.撇开技术上的问题,例如线程的实现和GIL,我觉得错误的教学指导才是主要问题. ...
- python实战===python控制键盘鼠标:pynput
Python控制键盘鼠标:pynput 地址:https://pypi.python.org/pypi/pynput 这个库让你可以控制和监控输入设备. 对于每一种输入设备,它包含一个子包来控制和监控 ...
- 06002001单例模式C#实现版本
书名:设计模式之禅 作者:秦小波 出版社:机械工业出版社 1 描述 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例 2 UML类图 图1-1 单例模式类图 3 代码 Singleto ...
- Spring3.0+Hibernate+Atomikos集成构建JTA的分布式事务--解决多数据源跨库事务
一.概念 分布式事务分布式事务是指事务的参与者.支持事务的服务器.资源服务器以及事务管理器分别位于不同的分布式系统的不同节点之上.简言之,同时操作多个数据库保持事务的统一,达到跨库事务的效果. JTA ...