2018.09.16 bzoj3757: 苹果树(树上莫队)
传送门
一道树上莫队。
先用跟bzoj1086一样的方法给树分块。
分完之后就可以莫队了。
但是两个询问之间如何转移呢?
感觉很难受啊。
我们定义S(u,v)" role="presentation" style="position: relative;">S(u,v)S(u,v)表示u->v这条路径的点集,T(u,v)" role="presentation" style="position: relative;">T(u,v)T(u,v)表示S(u,v)" role="presentation" style="position: relative;">S(u,v)S(u,v) xor" role="presentation" style="position: relative;">xorxor lca(u,v)" role="presentation" style="position: relative;">lca(u,v)lca(u,v),同时我们令上一个询问是S(predu,predv)" role="presentation" style="position: relative;">S(predu,predv)S(predu,predv)。
对于T(u,v)" role="presentation" style="position: relative;">T(u,v)T(u,v)有一个恒等式:
T(u,v)=S(root,u)" role="presentation" style="position: relative;">T(u,v)=S(root,u)T(u,v)=S(root,u) xor" role="presentation" style="position: relative;">xorxor S(root,v)" role="presentation" style="position: relative;">S(root,v)S(root,v)
因此有:
T(predu,predv)=S(root,predu)" role="presentation" style="position: relative;">T(predu,predv)=S(root,predu)T(predu,predv)=S(root,predu) xor" role="presentation" style="position: relative;">xorxor S(root,predv)" role="presentation" style="position: relative;">S(root,predv)S(root,predv)
因此有:
T(u,v)" role="presentation" style="position: relative;">T(u,v)T(u,v) xor" role="presentation" style="position: relative;">xorxor T(predu,predv)=S(root,u)" role="presentation" style="position: relative;">T(predu,predv)=S(root,u)T(predu,predv)=S(root,u) xor" role="presentation" style="position: relative;">xorxor S(root,v)" role="presentation" style="position: relative;">S(root,v)S(root,v) xor" role="presentation" style="position: relative;">xorxor S(root,predu)" role="presentation" style="position: relative;">S(root,predu)S(root,predu) xor" role="presentation" style="position: relative;">xorxor S(root,predv)" role="presentation" style="position: relative;">S(root,predv)S(root,predv)
因此有:
T(u,v)=T(predu,predv)" role="presentation" style="position: relative;">T(u,v)=T(predu,predv)T(u,v)=T(predu,predv) xor" role="presentation" style="position: relative;">xorxor T(predu,v)" role="presentation" style="position: relative;">T(predu,v)T(predu,v) xor" role="presentation" style="position: relative;">xorxor T(u,predv)" role="presentation" style="position: relative;">T(u,predv)T(u,predv)
那么更新就比较容易了。
代码:
#include<bits/stdc++.h>
#define N 100005
#define sig 24
using namespace std;
int n,m,rt,sum,ans[N],blo[N],dfn[N],dep[N],first[N],cnt,blos,top,fa[N][sig],mul[N],block,tot,stk[N],tim[N],col[N];
bool vis[N];
struct edge{int v,next;}e[N<<1];
struct Q{int u,v,a,b,id;}q[N];
inline bool cmp(Q a,Q b){return blo[a.u]==blo[b.u]?dfn[a.v]<dfn[b.v]:blo[a.u]<blo[b.u];}
inline void add(int u,int v){e[++cnt].v=v,e[cnt].next=first[u],first[u]=cnt;}
inline void dfs(int p){
dfn[p]=++tot,dep[p]=dep[fa[p][0]]+1;
int tmp=top;
for(int i=1;i<sig;++i)
if(dep[p]<mul[i])break;
else fa[p][i]=fa[fa[p][i-1]][i-1];
for(int i=first[p];i;i=e[i].next){
int v=e[i].v;
if(v==fa[p][0])continue;
fa[v][0]=p,dfs(v);
if(top-tmp>=block){
++blos;
while(top!=tmp)blo[stk[top--]]=blos;
}
}
stk[++top]=p;
}
inline int lca(int x,int y){
if(dep[x]<dep[y])x^=y,y^=x,x^=y;
int tmp=dep[x]-dep[y];
for(int i=0;i<sig;++i)if(tmp&(1<<i))x=fa[x][i];
if(x==y)return x;
for(int i=sig-1;~i;--i)if(fa[x][i]!=fa[y][i])x=fa[x][i],y=fa[y][i];
return fa[x][0];
}
inline void update(int x){
if(vis[x]){
vis[x]=0,--tim[col[x]];
if(tim[col[x]]==0)--sum;
}
else{
vis[x]=1,++tim[col[x]];
if(tim[col[x]]==1)++sum;
}
}
inline void reverse(int x,int y){
while(x!=y){
if(dep[x]<dep[y])x^=y,y^=x,x^=y;
update(x),x=fa[x][0];
}
}
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
int main(){
n=read(),m=read(),mul[0]=1,block=sqrt(n);
for(int i=1;i<=n;++i)col[i]=read();
for(int i=1;i<=n;++i){
int u=read(),v=read();
if(!u||!v)rt=u+v;
else add(u,v),add(v,u);
}
for(int i=1;i<sig;++i)mul[i]=mul[i-1]<<1;
dfs(rt),++blos;
while(top)blo[stk[top--]]=blos;
for(int i=1;i<=m;++i){
q[i].u=read(),q[i].v=read(),q[i].a=read(),q[i].b=read(),q[i].id=i;
if(dfn[q[i].u]>dfn[q[i].v])q[i].u^=q[i].v,q[i].v^=q[i].u,q[i].u^=q[i].v;
}
sort(q+1,q+m+1,cmp);
int Lca=lca(q[1].u,q[1].v);
reverse(q[1].u,q[1].v),update(Lca),ans[q[1].id]=sum;
if(q[1].a!=q[1].b&&tim[q[1].a]&&tim[q[1].b])--ans[q[1].id];
for(int i=2;i<=m;++i){
update(Lca),reverse(q[i-1].u,q[i].u),reverse(q[i-1].v,q[i].v);
Lca=lca(q[i].u,q[i].v),update(Lca),ans[q[i].id]=sum;
if(q[i].a!=q[i].b&&tim[q[i].a]&&tim[q[i].b])--ans[q[i].id];
}
for(int i=1;i<=m;++i)printf("%d\n",ans[i]);
return 0;
}
2018.09.16 bzoj3757: 苹果树(树上莫队)的更多相关文章
- [BZOJ3757]苹果树(树上莫队)
树上莫队共有三种写法: 1.按DFS序列分块,和普通莫队类似.常数大,不会被卡. 2.按块状树的方式分块.常数小,会被菊花图卡到O(n). 3.按[BZOJ1086]王室联邦的方式分块.常数小,不会被 ...
- 【BZOJ 3735】苹果树 树上莫队(树分块+离线莫队+鬼畜的压行)
2016-05-09 UPD:学习了新的DFS序列分块,然后发现这个东西是战术核导弹?反正比下面的树分块不知道要快到哪里去了 #include<cmath> #include<cst ...
- BZOJ.3757.苹果树(树上莫队)
题面链接 /* 代码正确性不保证..(不过交了SPOJ没WA T了最后一个点) 在DFS序做莫队 当一个点不是另一个点的LCA时,需要加上它们LCA的贡献 */ #include <cmath& ...
- 【BZOJ-3757】苹果树 块状树 + 树上莫队
3757: 苹果树 Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 1305 Solved: 503[Submit][Status][Discuss] ...
- BZOJ3757: 苹果树【树上莫队】
Description 神犇家门口种了一棵苹果树.苹果树作为一棵树,当然是呈树状结构,每根树枝连接两个苹果,每个苹果都可以沿着一条由树枝构成的路径连到树根,而且这样的路径只存在一条.由于这棵苹果树 ...
- 【BZOJ3757】苹果树(树上莫队)
点此看题面 大致题意: 每次问你树上两点之间路径中有多少种颜色,每次询问可能会将一种颜色\(a\)看成\(b\). 树上莫队 这题是一道树上莫队板子题. 毕竟求区间中有多少种不同的数是莫队算法的经典应 ...
- 树上莫队 wowow
构建:像线性的莫队那样,依旧是按sqrt(n)为一块分块. int dfs(int x){ ; dfn[x]=++ind; ;i<=;i++) if (bin[i]<=deep[x]) f ...
- 【BZOJ4129】Haruna’s Breakfast(树上莫队)
[BZOJ4129]Haruna's Breakfast(树上莫队) 题面 BZOJ Description Haruna每天都会给提督做早餐! 这天她发现早饭的食材被调皮的 Shimakaze放到了 ...
- 【SPOJ】Count On A Tree II(树上莫队)
[SPOJ]Count On A Tree II(树上莫队) 题面 洛谷 Vjudge 洛谷上有翻译啦 题解 如果不在树上就是一个很裸很裸的莫队 现在在树上,就是一个很裸很裸的树上莫队啦. #incl ...
随机推荐
- as3 判断鼠标移动方向
import flash.events.MouseEvent; var odx:Number=mouseX; var ody:Number=mouseY; stage.addEventListener ...
- EL&jsp
JSP 2.0(java server pages): EL 表达式 JSP九大内置对象及作用范围 JSP Directive JSP Action EL表达式: EL 算法(Arithmetic)表 ...
- 多媒体基础知识之PCM数据
1.什么是PCM音频数据 PCM(Pulse Code Modulation)也被称为脉冲编码调制.PCM音频数据是未经压缩的音频采样数据裸流,它是由模拟信号经过采样.量化.编码转换成的标准的数字音频 ...
- Shell常用命令find、grep总结
一.find 命令 find 命令的常用形式可以简化为 find [path...][expression] path:find 命令查找的目录路径,其中 ./ 表示当前目录,/ 表示系统根目录 ex ...
- 数据报表之Excel操作模块
Excel是当今最流行的电子表格处理软件,支持丰富的计算函数及图表,在系统运营方面广泛用于运营数据报表,比如业务质量.资源利用.安全扫描等报表,同时也是应用系统常见的文件导出格式,以便数据使用人员做进 ...
- 吴裕雄 数据挖掘与分析案例实战(5)——python数据可视化
# 饼图的绘制# 导入第三方模块import matplotlibimport matplotlib.pyplot as plt plt.rcParams['font.sans-serif']=['S ...
- python字符串填充(转)
ljust()方法返回字符串左对齐的字符串长度宽度.填充是通过使用指定的fillchar(默认为空格).如果宽度小于len(s)返回原始字符串.语法 以下是ljust()方法的语法: str.ljus ...
- python -m
影响sys.path python xxx.py python -m xxx.py 这是两种加载py文件的方式:1叫做直接运行2把模块当作脚本来启动 直接启动是把脚本所在的目录放到了sys.path属 ...
- java并发中的锁
java中的锁,最基本的是Lock接口. Lock接口中的方法,主要是: lock(): 获取锁,lock()方法会对Lock实例对象进行加锁,因此所有对该对象调用lock()方法的线程都会被阻塞,直 ...
- 第七章 二叉搜索树(b3)BST:删除