正题

题目链接:https://www.luogu.com.cn/problem/P6329


解题思路

给出\(n\)个点的一棵树,每个点有权值,有\(m\)次操作

  1. 修改一个点\(x\)的权值为\(y\)
  2. 询问距离点\(x\)不超过\(k\)的所有点点权和

解题思路

点分树的模板题,先点分治构造出点分树,然后在上面维护信息。

对于每个点维护一个点分子树内,与该点的距离为下标,点权为权值的的树状数组,然后查询的时候直接查距离不超过\(k-dis(now,x)\)的就好了。

发现与点分父节点会有算重的情况,这个时候顺便维护一个以与父节点的距离为下标的树状数组,然后减去重复的答案就好了。

时间复杂度\(O(n\log^2 n)\)。


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#define lowbit(x) (x&-x)
using namespace std;
const int N=1e5+10,T=18,inf=1e9;
struct node{
int to,next;
}a[N<<1];
int n,m,tot,cnt,num,root,fr,val[N];
int ls[N],f[N<<1][T],rfn[N],dep[N];
int fa[N],siz[N],lg[N<<1],mx;
bool v[N];
struct BIT{
vector<int> t;int n;
void Init(int x)
{x++;t.resize(x);n=x;return;}
void Change(int x,int val){
x++;
while(x<=n){
t[x-1]+=val;
x+=lowbit(x);
}
return;
}
int Ask(int x){
if(x<0)return 0;
int ans=0;x++;
if(x>n)x=n;
while(x){
ans+=t[x-1];
x-=lowbit(x);
}
return ans;
}
}s1[N],s2[N];
void addl(int x,int y){
a[++tot].to=y;
a[tot].next=ls[x];
ls[x]=tot;return;
}
void dfs(int x,int fa){
dep[x]=dep[fa]+1;
f[++cnt][0]=x;rfn[x]=cnt;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa)continue;
dfs(y,x);f[++cnt][0]=x;
}
return;
}
void groot(int x,int fa){
siz[x]=1;int f=0;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(v[y]||y==fa)continue;
groot(y,x);siz[x]+=siz[y];
f=max(f,siz[y]);
}
f=max(f,num-siz[x]);
if(f<fr)root=x,fr=f;
return;
}
void calc(int x,int fa,int dep){
mx=max(mx,dep);
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa||v[y])continue;
calc(y,x,dep+1);
}
return;
}
void Build(int x,int h){
v[x]=1;int S=num,z=fr;
mx=0;calc(x,x,0);
s1[x].Init(mx);
s2[x].Init(h);
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(v[y])continue;
num=(siz[y]>siz[x])?(S-siz[x]):siz[y];
mx=0;calc(y,x,1);
fr=inf;groot(y,x);y=root;fa[y]=x;
Build(y,mx);
}
return;
}
void Init(){
dfs(1,1);
for(int i=2;i<=cnt;i++)lg[i]=lg[i>>1]+1;
for(int j=1;(1<<j)<=cnt;j++)
for(int i=1;i+(1<<j)-1<=cnt;i++){
int x=f[i][j-1],y=f[i+(1<<j-1)][j-1];
f[i][j]=(dep[x]<dep[y])?x:y;
}
fr=inf;num=n;groot(1,1);
Build(root,0);
return;
}
int LCA(int l,int r){
l=rfn[l];r=rfn[r];
if(l>r)swap(l,r);
int z=lg[r-l+1];
int x=f[l][z],y=f[r-(1<<z)+1][z];
return (dep[x]<dep[y])?x:y;
}
int dis(int x,int y)
{return dep[x]+dep[y]-2*dep[LCA(x,y)];}
void Updata(int x,int val){
int now=x;
while(now){
s1[now].Change(dis(now,x),val);
if(fa[now])s2[now].Change(dis(fa[now],x),val);
now=fa[now];
}
return;
}
int Ask(int x,int k){
int ans=0,now=x;
ans=s1[x].Ask(k);
while(fa[now]){
int d=dis(fa[now],x);
ans+=s1[fa[now]].Ask(k-d);
if(fa[now])ans-=s2[now].Ask(k-d);
now=fa[now];
}
return ans;
}
int main()
{
// freopen("P6329_1.in","r",stdin);
// freopen("data.out","w",stdout);
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
scanf("%d",&val[i]);
for(int i=1;i<n;i++){
int x,y;
scanf("%d%d",&x,&y);
addl(x,y);addl(y,x);
}
Init();
for(int i=1;i<=n;i++)
Updata(i,val[i]);
int last=0;
while(m--){
int op,x,y;
scanf("%d%d%d",&op,&x,&y);
x^=last;y^=last;
if(op){
Updata(x,y-val[x]);
val[x]=y;
}
else
printf("%d\n",last=Ask(x,y));
}
return 0;
}

P6329-[模板]点分树 | 震波的更多相关文章

  1. 【题解】P6329 【模板】点分树 | 震波

    题外话 (其实模板题没必要在这里水题解的--主要是想说这个qwq) 小常数的快乐.jpg 我也不知道我为啥常数特别小跑得飞快--不加快读就能在 luogu 的最优解上跑到 rank5 ( 说不定深夜提 ...

  2. 一篇自己都看不懂的点分治&点分树学习笔记

    淀粉质点分治可真是个好东西 Part A.点分治 众所周知,树上分治算法有$3$种:点分治.边分治.链分治(最后一个似乎就是树链剖分),它们名字的不同是由于分治方式的不同的.点分治,顾名思义,每一次选 ...

  3. 线段树练习 3&&P3372 【模板】线段树 1

    题目描述 Description 给你N个数,有两种操作: 1:给区间[a,b]的所有数增加X 2:询问区间[a,b]的数的和. 输入描述 Input Description 第一行一个正整数n,接下 ...

  4. 【洛谷3345_BZOJ3924】[ZJOI2015]幻想乡战略游戏(点分树)

    大概有整整一个月没更博客了 -- 4 月为省选爆肝了一个月,最后压线进 B 队,也算给 NOIP2018 翻车到 316 分压线省一这个折磨了五个月的 debuff 画上了一个不算太差的句号.结果省选 ...

  5. 线段树学习笔记(基础&进阶)(一) | P3372 【模板】线段树 1 题解

    什么是线段树 线段树是一棵二叉树,每个结点存储需维护的信息,一般用于处理区间最值.区间和等问题. 线段树的用处 对编号连续的一些点进行修改或者统计操作,修改和统计的复杂度都是 O(log n). 基础 ...

  6. 洛谷P3372 【模板】线段树 1

    P3372 [模板]线段树 1 153通过 525提交 题目提供者HansBug 标签 难度普及+/提高 提交  讨论  题解 最新讨论 [模板]线段树1(AAAAAAAAA- [模板]线段树1 洛谷 ...

  7. 洛谷P3373 【模板】线段树 2

     P3373 [模板]线段树 2 47通过 186提交 题目提供者HansBug 标签 难度提高+/省选- 提交  讨论  题解 最新讨论 为啥WA(TAT) 题目描述 如题,已知一个数列,你需要进行 ...

  8. hdu 3074 Multiply game(模板级线段树)

    离机房关门还有十分钟,这点时间能干些什么?故作沉思地仰望星空,重新捋一下一天的学习进度,或者,砍掉一棵模板级线段树. 纯模板,就是把单点更新,区间求和改为单点更新,区间求积. 1A. #include ...

  9. Luogu3373【模板】线段树2

    P3373[模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格式: 第一行 ...

随机推荐

  1. sentinel使用(结合OpenFeign)

    前提 需要先安装sentinel. 父项目POM pom.xml <?xml version="1.0" encoding="UTF-8"?> &l ...

  2. C#基础知识---Linq操作XML文件

    概述 Linq也就是Language Integrated Query的缩写,即语言集成查询,是微软在.Net 3.5中提出的一项新技术. Linq主要包含4个组件---Linq to Objects ...

  3. JDBC高级篇(MYSQL)—— JDBC中初涉数据库事务

    注意:其中的JdbcUtil是我自定义的连接工具类:代码例子链接: package d_transaction; import java.sql.Connection; import java.sql ...

  4. mysql 用 source 导入备份数据

    输入命令:source E:/proDB/start2.sql;

  5. Go并发控制--WaitGroup篇

    目录 1. 前言 2. 使用WaitGroup控制 2.1 使用场景 2.2 信号量 1.3 WaitGroup 数据结构 2.3.1 Add () 方法 2.3.2 Wait() 2.3.3 Don ...

  6. 巧用模糊实现视觉的 3D 效果

    本文较短,将介绍巧用模糊实现视觉 3D 效果的技巧. 我们都知道,在正常的视觉效果中,离我们越近的通常我们会看的越清晰,而离我们较远则相对没那么清晰~ 我们可以利用清晰与模糊两种状态来构建视差效果.像 ...

  7. 简单C++线程池

    简单C++线程池 Java 中有一个很方便的 ThreadPoolExecutor,可以用做线程池.想找一下 C++ 的类似设施,尤其是能方便理解底层原理可上手的.网上找到的 demo,基本都是介绍的 ...

  8. 为开源项目 go-gin-api 增加后台任务模块

    目录 任务管理界面 (WEB) 任务调度器 任务执行器 小结 推荐阅读 任务管理界面 (WEB) 支持在 WEB 界面 中对任务进行管理,例如:新增任务.编辑任务.启用/禁用任务.手动执行任务 等. ...

  9. idea字节码插件JClassLib——阅读JVM字节码

    idea字节码插件JClassLib--阅读JVM字节码 生成字节码文件并查看 查看字节码文件的方式:javac 文件名.java 即可生成.class文件,但是这种方式不方便 java:是运行字节码 ...

  10. 从一个跨二十年的glibc bug说起

    1. 缘起 这几天调gcc 7.5.0 +glibc 2.23的交叉编译工具链,由于gcc 7.5.0的默认打开Werr,偶然发现了glibc一个隐藏了二十年的世纪大bug. 这个bug在glibc ...