bzoj1036 树的统计 树链剖分模板
题意:给出树上任意两点,求路径上的值的和与最大值,带单点修改操作
树链剖分思路:
1、对树进行dfs求出点的深度和父亲节点,然后求出轻重儿子(重儿子就是点最多的那个子树,其余都是轻儿子),用一个son数组指向每个节点的重儿子
2、对树进行第二次dfs,对于所有的重儿子,求出他的top节点也就是每个重儿子沿着重链可以到达的最远的那个祖先,然后维护dfs序,记录每个节点的访问次序以及第几次访问的是哪个节点,轻儿子的top节点就是本身
然后我们得到
dfs序: 1 4 9 13 14 8 10 3 7 2 6 11 12 5
top数组: 1 1 1 1 1 8 10 3 3 2 2 2 12 5(对应dfs序)
这样我们就把这棵树拆成了一条条的链(top值相同则为一条链上的点),用线段树维护这个dfs序,就可以快速求出链上最大值和值的和了,
对于任意两点,我们只需依次求出路径上的所有链的答案然后合并即可,可以证明路径上轻重链的条数是不超过logn的,这样单次查询的复杂度为O((logn)^2)
总时间复杂度O(q(logn)^2)
在计算两点答案的时候,采取一个巧妙的方法。首先若两个点不在同一条链上,我们总是让深度更大的那个点x往上跳到top[x],并统计这条链的答案,直到两个点到同一条链上,最后计算在一条链上时的答案即可
AC代码(模板)
#include<bits/stdc++.h>
using namespace std;
const int N = 1e5+;
struct Edge
{
int v,next;
}edge[N<<];
int sum[N<<],mx[N<<],n;
int head[N],tot,dep[N],fa[N],sz[N],son[N],top[N],id[N],rk[N],cnt,val[N];
void init()
{
memset(head,-, sizeof(head));
tot=;
}
void add(int u,int v)
{
edge[tot].v=v;
edge[tot].next=head[u];
head[u]=tot++;
}
void dfs1(int u,int f)
{
dep[u]=dep[f]+;
fa[u]=f;
sz[u]=;
for(int i=head[u];~i;i=edge[i].next)
{
int v=edge[i].v;
if(v==f)continue;
dfs1(v,u);
sz[u]+=sz[v];
if(sz[v]>sz[son[u]])son[u]=v;
}
}
void dfs2(int u,int t)
{
top[u]=t;
id[u]=++cnt;
rk[cnt]=u;
if(!son[u])return;
dfs2(son[u],t);
for(int i=head[u];~i;i=edge[i].next)
{
int v=edge[i].v;
if(v!=son[u]&&v!=fa[u])dfs2(v,v);
}
}
void pushup(int rt)
{
sum[rt]=sum[rt<<]+sum[rt<<|];
mx[rt]=max(mx[rt<<],mx[rt<<|]);
}
void build(int l,int r,int rt)
{
if(l==r)
{
mx[rt]=sum[rt]=val[rk[l]];
return;
}
int m=(l+r)>>;
build(l,m,rt<<);
build(m+,r,rt<<|);
pushup(rt);
}
int querySum(int L,int R,int l,int r,int rt)
{
if(L<=l&&r<=R)return sum[rt];
int m=(l+r)>>;
int res=;
if(L<=m)res+=querySum(L,R,l,m,rt<<);
if(R>m)res+=querySum(L,R,m+,r,rt<<|);
return res;
}
int queryMax(int L,int R,int l,int r,int rt)
{
if(L<=l&&r<=R)return mx[rt];
int m=(l+r)>>;
int res=-1e9;
if(L<=m)res=max(res,queryMax(L,R,l,m,rt<<));
if(R>m)res=max(res,queryMax(L,R,m+,r,rt<<|));
return res;
}
void update(int pos,int val,int l,int r,int rt)
{
if(l==r){
sum[rt]=mx[rt]=val;
return;
}
int m=(l+r)>>;
if(pos<=m)update(pos,val,l,m,rt<<);
else update(pos,val,m+,r,rt<<|);
pushup(rt);
}
int getSum(int x,int y)
{
int ans=;
while(top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]])swap(x,y);
ans+=querySum(id[top[x]],id[x],,n,);
x=fa[top[x]];
}
if(id[x]>id[y])swap(x,y);
ans+=querySum(id[x],id[y],,n,);
return ans;
}
int getMax(int x,int y)
{
int ans=-1e9;
while (top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]])swap(x,y);
ans=max(ans,queryMax(id[top[x]],id[x],,n,));
x=fa[top[x]];
}
if(id[x]>id[y])swap(x,y);
ans=max(ans,queryMax(id[x],id[y],,n,));
return ans;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie();
cout.tie();
init();
int u,v;
cin>>n;
for(int i=;i<=n-;i++)
{
cin>>u>>v;
add(u,v);
add(v,u);
}
for(int i=;i<=n;i++)cin>>val[i];
dfs1(,);
dfs2(,);
build(,n,);
int q,x,y;
string op;
cin>>q;
while(q--)
{
cin>>op>>x>>y;
if(op=="QMAX")cout<<getMax(x,y)<<'\n';
else if(op=="QSUM")cout<<getSum(x,y)<<'\n';
else update(id[x],y,,n,);
}
return ;
}
bzoj1036 树的统计 树链剖分模板的更多相关文章
- 树的统计Count---树链剖分
NEFU专项训练十和十一——树链剖分 Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t ...
- BZOJ 1036: [ZJOI2008]树的统计Count-树链剖分(点权)(单点更新、路径节点最值、路径求和)模板,超级认真写了注释啊啊啊
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 23015 Solved: 9336[Submit ...
- 算法复习——树链剖分模板(bzoj1036)
题目: 题目背景 ZJOI2008 DAY1 T4 题目描述 一棵树上有 n 个节点,编号分别为 1 到 n ,每个节点都有一个权值 w .我们将以下面的形式来要求你对这棵树完成一些操作:I.CHAN ...
- BZOJ 2243 染色 | 树链剖分模板题进阶版
BZOJ 2243 染色 | 树链剖分模板题进阶版 这道题呢~就是个带区间修改的树链剖分~ 如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了. 这道题要注意的是,无论是线段树上 ...
- Hdu 5274 Dylans loves tree (树链剖分模板)
Hdu 5274 Dylans loves tree (树链剖分模板) 题目传送门 #include <queue> #include <cmath> #include < ...
- [luogu P2590 ZJOI2008] 树的统计 (树链剖分)
题目描述 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w. 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u ...
- bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题
[ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...
- BZOJ-1036 树的统计Count 链剖线段树(模板)=(树链剖分+线段树)
潇爷昨天刚刚讲完...感觉得还可以...对着模板打了个模板...还是不喜欢用指针.... 1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Lim ...
- BZOJ 1036: [ZJOI2008]树的统计Count (树链剖分模板题)
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 14982 Solved: 6081[Submit ...
随机推荐
- 去BAT面试完的Mysql面试题总结(55道,带完整答案)
1.一张表里面有ID自增主键,当insert了17条记录之后,删除了第15,16,17条记录,再把mysql重启,再insert一条记录,这条记录的ID是18还是15 ? 2.mysql的技术特点是什 ...
- HashSet、LinkedHashSet、TreeSet的区别
HashSet:哈希表是通过使用称为散列法的机制来存储信息的,元素并没有以某种特定顺序来存放: LinkedHashSet:以元素插入的顺序来维护集合的链接表,允许以插入的顺序在集合中迭代: Tree ...
- shell位置参数和 shift 命令
- ZOJ-3662 Math Magic 背包DP
这题不错,可惜我还是太弱了,没想到qwq. 看了网上大佬题解之后写的,对比了一下代码,好像我写的还是挺简洁的(逃,只是吞行比较多). 因为直接用lcm的值做下标会超时,所以我们观察发现可以组成lcm为 ...
- day08 python文件操作
day08 python 一.文件操作 1.文件操作的函数 open(文件名, mode=模式, encoding=字符集) 2.模式: r, w, a, r+ ...
- 对malloc与free函数的浅识
本文介绍malloc和free函数的内容. 在C中,对内存的管理是相当重要.下面开始介绍这两个函数: 一.malloc()和free()的基本概念以及基本用法: .函数原型及说明: void *mal ...
- PHP-在排序数组中查找元素的第一个和最后一个位置
给定一个按照升序排列的整数数组 nums,和一个目标值 target.找出给定目标值在数组中的开始位置和结束位置. 你的算法时间复杂度必须是 O(log n) 级别. 如果数组中不存在目标值,返回 [ ...
- Eclipse导入的Maven项目没有Build Path
我导入的是 Signal-Server项目到 Eclipse中,发现 src 文件夹上面没有#号,包视图和语法提示都没有 ~~ 解决方法: 修改 Project Facets 在 项目右键 -> ...
- 重视项目排期,对dateline 有所敬畏
项目排期 = 个人任务完成 + 风险预估 + 意外情况 + 项目上下游依赖 个人任务完成 个人任务具体话 不要考虑私人情感,专注工作 风险预估 对可能出现的情况进行考虑 意外情况 对出现的意外情况提前 ...
- selenium基本元素定位-findElement(By.*)
selenium基本元素的定位和操作 一. 查找元素 1.1 findElement(By.id) // by——>定位器——>以某种方式去找元素 driver.findElement(b ...