BZOJ3730 震波 【动态点分治】*
BZOJ3730 震波
Description
在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i]。
不幸的是,这片土地常常发生地震,并且随着时代的发展,城市的价值也往往会发生变动。
接下来你需要在线处理M次操作:
0 x k 表示发生了一次地震,震中城市为x,影响范围为k,所有与x距离不超过k的城市都将受到影响,该次地震造成的经济损失为所有受影响城市的价值和。
1 x y 表示第x个城市的价值变成了y。
为了体现程序的在线性,操作中的x、y、k都需要异或你程序上一次的输出来解密,如果之前没有输出,则默认上一次的输出为0。
Input
第一行包含两个正整数N和M。
第二行包含N个正整数,第i个数表示value[i]
接下来N-1行,每行包含两个正整数u、v,表示u和v之间有一条无向边。
接下来M行,每行包含三个数,表示M次操作。
Output
包含若干行,对于每个询问输出一行一个正整数表示答案。
Sample Input
8 1
1 10 100 1000 10000 100000 1000000 10000000
1 2
1 3
2 4
2 5
3 6
3 7
3 8
0 3 1
Sample Output
11100101
HINT
1<=N,M<=100000
1<=u,v,x<=N
1<=value[i],y<=10000
0<=k<=N-1
动态点分治
噗噗噗恶习死了
首先我们是肯定需要求出两点间的路径的,然后就可以上倍增LCA的板子
然后开始考虑怎么维护答案
如果不考虑修改
每次询问的答案可以用怎样的形式来统计呢?
因为我们维护一个点的字树信息很方便,所以就可以先用树状数组维护一下一个子树里的信息,以距离为下标,点权直接单点加上就行了
现在考虑怎么统计子树外边的贡献,首先我们可以跳到点分树上的prt节点上,然后求出prt到当前节点u的距离len,用查询距离k减去len,查询在prt的子树里边有多少距离小于等于k-len的点就可以了,但是我们发现在u的子树内的一部分节点会被重复计算,所以我们需要另开一个树状数组来维护重复的贡献,在这个树状数组里边只记录在u的子树中的额外贡献
现在考虑吧修改加上,我们从修改节点x这个点开始,不停向prt跳,然后我们这里可以把每一个原来是val[x]的贡献换成y的贡献,简而言之就是加上y-val[x],然后跟查询差不多,只不过反向维护一下就好了
然后细节一大堆,主要就是跳prt维护的细节
#include<bits/stdc++.h>
using namespace std;
#define N 500010
inline int read(){
int ans=0,w=1;char c=getchar();
while(!isdigit(c)&&c!='-')c=getchar();
if(c=='-')w=-1,c=getchar();
while(isdigit(c))ans=(ans<<1)+(ans<<3)+c-'0',c=getchar();
return ans*w;
}
int n,m,lastans=0,val[N],Log[N];
struct Edge{
int v,next;
Edge(){}
Edge(int v,int next):v(v),next(next){}
}E[N<<1];
int head[N],tot=0;
void add(int u,int v){
E[++tot]=Edge(v,head[u]);
head[u]=tot;
}
int dep[N]={0},Fa[N][21];
void getFa(int u,int fa){
Fa[u][0]=fa;
dep[u]=dep[fa]+1;
for(int i=1;i<=20;i++)Fa[u][i]=Fa[Fa[u][i-1]][i-1];
for(int i=head[u];i;i=E[i].next){
int v=E[i].v;
if(v==fa)continue;
getFa(v,u);
}
}
int Lca(int x,int y){
if(dep[x]<dep[y])swap(x,y);
int t=dep[x]-dep[y],k=Log[n]+1;
for(int i=0;i<=k;i++)
if(t&(1<<i))x=Fa[x][i];
if(x==y)return x;
while(Fa[x][0]!=Fa[y][0]){
if(Fa[x][k]!=Fa[y][k]){
x=Fa[x][k];
y=Fa[y][k];
}
k--;
}
return Fa[x][0];
}
int getdis(int x,int y){return dep[x]+dep[y]-dep[Lca(x,y)]*2;}
vector<int> t[2][N];
void modify(int typ,int pos,int x,int vl){
if(!x)return;
int up=t[typ][pos].size();
while(x<up){
t[typ][pos][x]+=vl;
x+=x&(-x);
}
}
int query(int typ,int pos,int x){
int res=0;
x=min(x,(int)(t[typ][pos].size()-1));
while(x){
res+=t[typ][pos][x];
x-=x&(-x);
}
return res;
}
int siz[N],F[N],rt,siz_tree;
int prt[N];bool vis[N];
void getroot(int u,int fa){
siz[u]=1;F[u]=0;
for(int i=head[u];i;i=E[i].next){
int v=E[i].v;
if(v==fa||vis[v])continue;
getroot(v,u);
siz[u]+=siz[v];
F[u]=max(F[u],siz[v]);
}
F[u]=max(F[u],siz_tree-siz[u]);
if(F[u]<F[rt])rt=u;
}
void divide(int u,int fa){
prt[u]=fa;vis[u]=1;
t[0][u].resize(siz_tree+1);
t[1][u].resize(siz_tree+1);
for(int i=head[u];i;i=E[i].next){
int v=E[i].v;
if(vis[v])continue;
F[rt=0]=siz_tree=siz[v];
getroot(v,u);
divide(rt,u);
}
}
void modify(int u,int vl){
for(int i=u;i;i=prt[i]){
modify(0,i,getdis(u,i),vl);
if(!prt[i])break;
modify(1,i,getdis(u,prt[i]),vl);
}
}
int query(int u,int vl){
int res=query(0,u,vl)+val[u];
for(int i=prt[u],last=u;i;last=i,i=prt[i]){
int d=vl-getdis(u,i);
if(d<0)continue;
res+=query(0,i,d)-query(1,last,d)+val[i];
}
return res;
}
void work(){
F[rt=0]=siz_tree=n;
getroot(1,0);
divide(rt,0);
}
int main(){
freopen("bzoj3730.in","r",stdin);
Log[0]=-1;for(int i=1;i<N;i++)Log[i]=Log[i>>1]+1;
n=read();m=read();
for(int i=1;i<=n;i++)val[i]=read();
for(int i=1;i<n;i++){
int u=read(),v=read();
add(u,v);
add(v,u);
}
getFa(1,0);
work();
for(int i=1;i<=n;i++)modify(i,val[i]);
for(int i=1;i<=m;i++){
int op=read(),x=read(),y=read();
x^=lastans,y^=lastans;
if(op)modify(x,y-val[x]),val[x]=y;
else printf("%d\n",lastans=query(x,y));
}
return 0;
}
BZOJ3730 震波 【动态点分治】*的更多相关文章
- bzoj3730 震波 [动态点分治,树状数组]
传送门 思路 如果没有强制在线的话可以离线之后CDQ分治随便搞. 有了强制在线之后--可能可以二维线段树?然而我不会算空间. 然后我们莫名其妙地想到了动态点分治,然后这题就差不多做完了. 点分树有一个 ...
- BZOJ3730震波——动态点分治+线段树(点分树套线段树)
题目描述 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土地常常发生地震,并且随着时代的发展,城市 ...
- BZOJ3730 震波 | 动态点分治
#include <cstdio> #include <cstring> #include <cmath> #include <algorithm> # ...
- bzoj3730 [震波][动态树分治+线段树+LCA]
震波 Time Limit: 15 Sec Memory Limit: 256 MBSubmit: 1573 Solved: 358[Submit][Status][Discuss] Descri ...
- 【BZOJ3730】震波 动态树分治+线段树
[BZOJ3730]震波 Description 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土 ...
- 【BZOJ-3730】震波 动态点分治 + 树状数组
3730: 震波 Time Limit: 15 Sec Memory Limit: 256 MBSubmit: 626 Solved: 149[Submit][Status][Discuss] D ...
- 【bzoj3730】震波 动态点分治+线段树
题目描述 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土地常常发生地震,并且随着时代的发展,城市 ...
- 【BZOJ3730】震波 - 动态点分治
题意: Description 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i]. 不幸的是,这片土地常常发生地震, ...
- bzoj 3730: 震波 动态点分治_树链剖分_线段树
##### 题目描述 : 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土地常常发生地震,并且随着 ...
- BZOJ 4372/3370 烁烁的游戏/震波 (动态点分治+线段树)
烁烁的游戏 题目大意: 给你一棵$n$个节点的树,有$m$次操作,询问某个节点的权值,或者将与某个点$x$距离不超过$d$的所有节点的权值都增加$w$ 动态点分裸题 每个节点开一棵权值线段树 对于修改 ...
随机推荐
- 问下大家,chorme里用开发者工具看headers,点network标签然后刷新网页并没有headers选项,怎么破?
问下大家,chorme里用开发者工具看headers,点network标签然后刷新网页并没有headers选项,怎么破? 请教个问题 jmeter在Linux服务器压测,抛出很多错误率 但日志中没看到 ...
- mysql的基本的数据库的查询
学习一个数据库我们要学习哪些东西: sql数据库的话, curd. 对于查询,要注意表的关联的查询. 索引,触发器,对于控制连接量,脚本, 数据库的可视化工具,权限管理. http://www.360 ...
- spring mvc: 属性方法名称解析器(多动作控制器)MultiActionController/ControllerClassNameHandlerMapping/PropertiesMethodNameResolver
spring mvc: 属性方法名称解析器(多动作控制器) 加入控制器是StudentContrller.java,里面有3个方法 index,add,remove 那么访问地址是: http://l ...
- Dapper 条件语句(Where) 中参数使用
public static List<ECInput> GetECInputList(DateTime beginDate,DateTime endDate,string[] barcod ...
- 搞懂分布式技术9:Nginx负载均衡原理与实践
搞懂分布式技术9:Nginx负载均衡原理与实践 本篇摘自<亿级流量网站架构核心技术>第二章 Nginx负载均衡与反向代理 部分内容. 当我们的应用单实例不能支撑用户请求时,此时就需要扩容, ...
- find命令中选项-path和-prune的使用
在Windows中可以在某些路径中查找文件,也可以设定不在某些路径中查找文件,下面用Linux中的find的命令结合其-path -prune参数来看看在Linux中怎么实现此功能.假如在当前目录下查 ...
- garylog学习篇
官方文档:https://www.graylog.org/ 简介 Graylog 是一个简单易用.功能较全面的日志管理工具,相比 ELK 组合, 优点: 部署维护简单查询语法简单易懂(对比ES的语法… ...
- Spring入门3.AOP编程
Spring入门3.AOP编程 代码下载: 链接: http://pan.baidu.com/s/11mYEO 密码: x7wa 前言: 前面学习的知识是Spring在Java项目中的IoC或DJ,这 ...
- 转载:【Oracle 集群】RAC知识图文详细教程(七)--Oracle 11G RAC集群安装
文章导航 集群概念介绍(一) ORACLE集群概念和原理(二) RAC 工作原理和相关组件(三) 缓存融合技术(四) RAC 特殊问题和实战经验(五) ORACLE 11 G版本2 RAC在LINUX ...
- 极简MarkDown教程(常用样式)
推荐编辑软件,NotePad++ & MarkDownViewer++(插件),以下内容为MarkDown格式,可自行放到编辑软件中查看,或在线查看 #### . 标题 > 用#+空格开 ...