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 ...
随机推荐
- win2008 server ping不同
win2008 server ping不同,网络正常. 下图可以解决!!!
- JPQL和SQL的比较
前言 在JAVA EE中,JPQL是专门为Java 应用程序访问和导航实体实例设计的.Java Presistence Query Language(JPQL),java持久性查询语言.它是JPA规范 ...
- spring data jpa分页5种方法
7.12.12 分页 本地sql查询 注意表名啥的都用数据库中的名称, 适用于特定数据库的查询 public interface UserRepository extends JpaRepositor ...
- Python之reduce函数
描述 reduce() 函数会对参数序列中元素进行累积. 函数将一个数据集合(链表,元组等)中的所有数据进行下列操作:用传给reduce中的函数 function(有两个参数)先对集合中的第 1.2 ...
- javascript中 try catch用法
javascript中 try catch用法 投稿:hebedich 字体:[增加 减小] 类型:转载 时间:2015-08-16我要评论 JS try catch语句一般在什么情况下使用?是必须使 ...
- 转载:canal数据库同步redis
ref: http://blog.csdn.net/tb3039450/article/details/53928351
- Swift - JPush极光推送的使用3(根据Alias别名,给某个指定用户发推送)(转)
一.别名(alias)介绍 (1)我们可以给每一个安装了应用程序的用户,取不同别名来标识(比如可以使用用户账号的 userid 来作为别名). (2)以后给某个特定用户推送消息时,就可以用此别名来指定 ...
- Spring MVC国际化
本文基于Spring MVC 注解-让Spring跑起来.本文提到的国际化是Spring实现国际化的方案之一. (1) 在applicationContext.xml中添加以下配置信息: <!- ...
- Docker之My sql 之旅
要用到mysql 数据库,本来想在本机装,后来想想还是有点污染环境,既然有docker为什么不用呢? 于是开启了采坑之旅,与其说采坑,倒不如说看文档不仔细看. docker pull mysql:5. ...
- js原生实现 无缝滚动图片
<!-- 无缝滚动图片开始 --> <div id=demopic style="OVERFLOW: hidden; WIDTH: 100%; COLOR: #fff; H ...