题目大意:

有一棵有根树,根为 1 ,点有点权。
现在有 m 次操作,操作有 3 种:
1 x y w ,将 x 到 y 的路径上的点点权加上 w (其中 w=±1w=±1 );
2 x y ,询问在 x 到 y 的路径上有多少个点点权 >0 ;
3 x ,询问在 x 的子树里的点有多少个点点权 >0 。

数据范围:$n,m≤10^5$,点权的绝对值$≤10^9$。

这一题正常的做法并不是特别优秀,我们考虑一些分块做法

考虑求一个连续的区间内有多少个数$>0$,我们显然可以将原序列分成$\sqrt{n}$块,对于每一个块我们开一个桶存储块内所有数,然后类似前缀和的方式扫一遍,查询一个块就可以$O(1)$实现查询了

修改怎么做?

首先是整个块一起变大/变小的情况,我们对整一块打一个标记$tag[x]$,下次查询时不要查询$>0$的,改为查询块内$>tag[x]$的数的数量。

某个块更改一部分的情况:我们修改原数,然后直接更改该块对应的桶即可,详见代码。

这个做法既然可以针对某一个区间,且资瓷修改。

回到原先的问题中,我们对给出的树树剖一下,每条链我们都分块维护一下,在此不再赘述。

查询某条路径的话,我们将询问拆成$\log{n}$条链,分别查询然后加起来即可。

查询某个子树内的话,直接查就可以了。

时间复杂度:$O(n^{1.5}\log\ n)$,代码只要3k。

别想什么树分块!!!

 #include<bits/stdc++.h>
#define L long long
#define M 100005
#define N 500
using namespace std; int B[M]={},S[M]={},tag[M]={},E[M]={},pos[M]={},a[M]={};short num[M/N+][M*]={};
void upd(short x[],int &y,int Val){if(Val==) ++x[++y+];else --x[y--+];}
void updata(int l,int r,int w){
if(B[l]==B[r]){for(int i=l;i<=r;i++) upd(num[B[i]],a[i],w); return;}
for(int i=l;B[i]==B[l];i++) upd(num[B[i]],a[i],w);
for(int i=r;B[i]==B[r];i--) upd(num[B[i]],a[i],w);
for(int i=B[l]+;i<B[r];i++) tag[i]+=w;
}
L query(int l,int r){
L res=;
if(B[l]==B[r]){for(int i=l;i<=r;i++) res+=(a[i]+tag[B[i]]>); return res;}
for(int i=l;B[i]==B[l];i++) res+=(a[i]+tag[B[i]]>);
for(int i=r;B[i]==B[r];i--) res+=(a[i]+tag[B[i]]>);
for(int i=B[l]+;i<B[r];i++) res+=num[i][-tag[i]];
return res;
} struct edge{int u,next;}e[M*]={}; int head[M]={},use=;
void add(int x,int y){use++;e[use].u=y;e[use].next=head[x];head[x]=use;}
int siz[M]={},son[M]={},dep[M]={},fa[M]={},dfn[M]={},low[M]={},top[M]={},t=;
void dfs(int x){
siz[x]=; dep[x]=dep[fa[x]]+;
for(int i=head[x];i;i=e[i].next) if(e[i].u!=fa[x]){
fa[e[i].u]=x; dfs(e[i].u);
siz[x]+=siz[e[i].u];
if(siz[son[x]]<siz[e[i].u]) son[x]=e[i].u;
}
}
void dfs(int x,int Top){
top[x]=Top; dfn[x]=++t;
if(son[x]) dfs(son[x],Top);
for(int i=head[x];i;i=e[i].next) if(e[i].u!=fa[x]&&e[i].u!=son[x]) dfs(e[i].u,e[i].u);
low[x]=t;
} L Query(int x,int y){
L res=;
for(;top[x]!=top[y];x=fa[top[x]]){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
res+=query(dfn[top[x]],dfn[x]);
}
res+=query(min(dfn[x],dfn[y]),max(dfn[x],dfn[y]));
return res;
}
void Updata(int x,int y,int w){
for(;top[x]!=top[y];x=fa[top[x]]){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
updata(dfn[top[x]],dfn[x],w);
}
updata(min(dfn[x],dfn[y]),max(dfn[x],dfn[y]),w);
} #define MFLONG 18000000
#define NUM(x) ((48<=x&&x<=57)||x=='-')
char _c[MFLONG];int _ns=,_nw=;int _x[],_ld;
inline void rd(int &_q){int _fu;if(_c[_ns]==) return;while(!NUM(_c[_ns])) _ns++;if(_c[_ns]=='-') _fu=-,_ns++;else _fu=;_q=;while(NUM(_c[_ns])) _q=_q*+_c[_ns++]-;_q=_fu*_q;} int n,m,T,ans=;
int main(){
fread(_c,,MFLONG,stdin);
memset(S,,sizeof(S));
rd(n); rd(m); rd(T);
for(int i=;i<=n;i++){
B[i]=i/N+;
S[B[i]]=min(S[B[i]],i);
E[B[i]]=max(E[B[i]],i);
}
for(int i=,x,y;i<n;i++) rd(x),rd(y),add(x,y),add(y,x);
dfs(); dfs(,);
for(int i=;i<=n;i++){
rd(a[dfn[i]]);
if(a[dfn[i]]>1e5) a[dfn[i]]=1e5;
if(a[dfn[i]]<-1e5) a[dfn[i]]=-1e5;
}
for(int x=;x<=B[n];x++){
for(int i=S[x];i<=E[x];i++) ++num[x][a[i]+];
for(int j=;~j;j--) num[x][j]+=num[x][j+];
}
while(m--){
int op,x,y,w; rd(op); rd(x); x^=T*ans;
if(op==) {printf("%lld\n",ans=query(dfn[x],low[x])); continue;}
rd(y); y^=T*ans;
if(op==) {printf("%lld\n",ans=Query(x,y)); continue;}
rd(w); Updata(x,y,w);
}
}

【UOJ#435】【集训队作业2018】Simple Tree 分块+树链剖分的更多相关文章

  1. [集训队作业2018]蜀道难——TopTree+贪心+树链剖分+链分治+树形DP

    题目链接: [集训队作业2018]蜀道难 题目大意:给出一棵$n$个节点的树,要求给每个点赋一个$1\sim n$之内的权值使所有点的权值是$1\sim n$的一个排列,定义一条边的权值为两端点权值差 ...

  2. Codeforces Round #329 (Div. 2) D. Happy Tree Party 树链剖分

    D. Happy Tree Party Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/593/p ...

  3. 【POJ3237】Tree(树链剖分)

    题意:在一棵N个节点,有边权的树上维护以下操作: 1:单边修改,将第X条边的边权修改成Y 2:区间取反,将点X与Y在树上路径中的所有边边权取反 3:区间询问最大值,询问X到Y树上路径中边权最大值 n& ...

  4. HDU 4718 The LCIS on the Tree(树链剖分)

    Problem Description For a sequence S1, S2, ... , SN, and a pair of integers (i, j), if 1 <= i < ...

  5. POJ 3237:Tree(树链剖分)

    http://poj.org/problem?id=3237 题意:树链剖分.操作有三种:改变一条边的边权,将 a 到 b 的每条边的边权都翻转(即 w[i] = -w[i]),询问 a 到 b 的最 ...

  6. QTREE3 spoj 2798. Query on a tree again! 树链剖分+线段树

    Query on a tree again! 给出一棵树,树节点的颜色初始时为白色,有两种操作: 0.把节点x的颜色置反(黑变白,白变黑). 1.询问节点1到节点x的路径上第一个黑色节点的编号. 分析 ...

  7. spoj 375 Query on a tree(树链剖分,线段树)

      Query on a tree Time Limit: 851MS   Memory Limit: 1572864KB   64bit IO Format: %lld & %llu Sub ...

  8. bzoj 3637: Query on a tree VI 树链剖分 && AC600

    3637: Query on a tree VI Time Limit: 8 Sec  Memory Limit: 1024 MBSubmit: 206  Solved: 38[Submit][Sta ...

  9. poj 3237 Tree(树链剖分,线段树)

    Tree Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 7268   Accepted: 1969 Description ...

随机推荐

  1. 【Web】网站主如何更改网页标签的图标(favicon.ico)

    修改web项目的favicon图标,方式有两种:全局方式和局部方式 全局方式: 进入服务器\webapps\ROOT,然后用自己的favicon.ico替换服务器自带的favicon.ico图片 局部 ...

  2. 《Just for Fun》---读后感

    <Just for Fun>本书是Linux之父林纳斯自传,书名的意思是:只是为了好玩.主要是讲了林纳斯的人生经历,以及Linux的诞生过程.Linux从一个终端仿真器到一个世界瞩目的操作 ...

  3. 学习Java的方法

    许多人在刚开始学习Java时,会因为学习方法的不正确,而丧失信心,从而半途而废.所以,今天,巩固就要教教大家学习Java的方法. 1.多练习 编程其实是一个非常抽象的东西,要想学好它,就不能只是看看书 ...

  4. 一个 图片 滚动 飞入的css特效

    @keyframes bounceInLeft { from, 60%, 75%, 90%, to {animation-timing-function: cubic-bezier(0.215, 0. ...

  5. Django介绍(2)

    https://www.cnblogs.com/yuanchenqi/articles/5658455.html

  6. Linux IPC之共享内存

    System V共享内存机制: shmget  shmat  shmdt  shmctl 原理及实现: system V IPC机制下的共享内存本质是一段特殊的内存区域,进程间需要共享的数据被放在该共 ...

  7. Struts has detected an unhandled exception

    这个问题是struts和jsp页面的配置之间产生了问题,就是struts里边的名字和jsp里用的名字不是同一个名字,所以无法识别,所以检查一下里边的命名.

  8. x+y+z=n的正整数解

    题:x+y+z=n,其中(n>=3),求x,y,z的正整数解的个数根据图象法:x>=1,y>=1,x+y<=n-1

  9. 201709025工作日记--更新UI方法

    1.handler+Thread 和 runOnUIThread 和 handler.post 方法 区别: 从实现原理上,两者别无二致,runOnUiThread也是借助Handler实现的.  对 ...

  10. Tensorflow currently has no official prebuild for your CUDA, cuDNN combination.

    INFO CUDA version: 10. ERROR cuDNN not found. See https://github.com/deepfakes/faceswap/blob/master/ ...