正题

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


题目大意

\(n\)个点的一棵树\(m\)次询问树上颜色。

强制在线

\(1\leq n\leq 4\times 10^5,1\leq m\leq 10^5,0\leq val_i<2^{31}\)


解题思路

把所有深度为\(\sqrt n\)并且下面至少有\(\sqrt n\)的深度的点标记,这样保证关键点数量不超过\(\sqrt n\)。

然后每个点到他周围关键点的距离也不会超过 \(\sqrt n\) 。

这样可以处理出关键点两两之间的颜色\(bitset\),然后每次路径找两个最近的关键点爆做就好了。

时间复杂度\(O((m+n)(\sqrt n+\frac{n}{\omega}))\)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<bitset>
#include<cmath>
using namespace std;
const int N=41000,T=300,M=(4e4)/T+10;
struct node{
int to,next;
}a[N<<1];
int n,m,tot,cnt,sum,fa[N],v[N],top[N],ls[N],d[N],dep[N],w[N],b[N],mark[N],dfn[N],ed[N],g[M][M];
bitset<N> f[M][M],bt;
void addl(int x,int y){
a[++tot].to=y;
a[tot].next=ls[x];
ls[x]=tot;return;
}
void dfs(int x){
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa[x])continue;
dep[y]=dep[x]+1;
fa[y]=x;dfs(y);
d[x]=max(d[x],d[y]+1);
}
if(dep[x]%T==0&&d[x]>=T)
mark[x]=++cnt;
return;
}
void dFs(int x){
if(mark[x])top[x]=x;dfn[x]=++cnt;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa[x])continue;
top[y]=top[x];dFs(y);
}
ed[x]=cnt;
return;
}
void calc(int x,int p,int fa){
if(!v[w[x]])bt[w[x]]=1,sum++;v[w[x]]++;
if(mark[x])f[p][mark[x]]=bt,g[p][mark[x]]=sum;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa)continue;
calc(y,p,x);
}
v[w[x]]--;if(!v[w[x]])bt[w[x]]=0,sum--;
return;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
scanf("%d",&w[i]),b[i]=w[i];
sort(b+1,b+1+n);
int L=unique(b+1,b+1+n)-b-1;
for(int i=1;i<=n;i++)
w[i]=lower_bound(b+1,b+1+L,w[i])-b;
for(int i=1;i<n;i++){
int x,y;
scanf("%d%d",&x,&y);
addl(x,y);addl(y,x);
}
dfs(1);cnt=0;dFs(1);
for(int i=1;i<=n;i++)
if(mark[i])calc(i,mark[i],i);
int last=0;
while(m--){
int x,y;
scanf("%d%d",&x,&y);
x^=last;
if(top[x]==top[y]){
int xx=x,yy=y,ans=0;
while(x!=y){
if(dep[x]<dep[y])swap(x,y);
if(!v[w[x]])ans++,v[w[x]]=1;
x=fa[x];
}
if(!v[w[x]])ans++,v[w[x]]=1;
printf("%d\n",ans);last=ans;
x=xx;y=yy;
while(x!=y){
if(dep[x]<dep[y])swap(x,y);
v[w[x]]=0;x=fa[x];
}
v[w[x]]=0;
}
else{
if(dfn[x]>=dfn[y]&&dfn[x]<=ed[y])swap(x,y);
if(dfn[y]>=dfn[x]&&dfn[y]<=ed[x]){
int z=top[y];
while(top[fa[z]]!=top[x])
z=top[fa[z]];
bt=f[mark[z]][mark[top[y]]];
int ans=g[mark[z]][mark[top[y]]];
while(y!=top[y]){
if(!bt[w[y]])ans++,bt[w[y]]=1;
y=fa[y];
}
while(z!=x){
if(!bt[w[z]])ans++,bt[w[z]]=1;
z=fa[z];
}
if(!bt[w[z]])ans++,bt[w[z]]=1;
printf("%d\n",ans);last=ans;
}
else{
bt=f[mark[top[x]]][mark[top[y]]];
int ans=g[mark[top[x]]][mark[top[y]]];
while(x!=top[x]){
if(!bt[w[x]])ans++,bt[w[x]]=1;
x=fa[x];
}
while(y!=top[y]){
if(!bt[w[y]])ans++,bt[w[y]]=1;
y=fa[y];
}
printf("%d\n",ans);last=ans;
}
}
}
return 0;
}

P6177-Count on a tree II/[模板]树分块的更多相关文章

  1. 洛谷 P6177 - Count on a tree II/【模板】树分块(树分块)

    洛谷题面传送门 好家伙,在做这道题之前我甚至不知道有个东西叫树分块 树分块,说白了就是像对序列分块一样设一个阈值 \(B\),然后在树上随机撒 \(\dfrac{n}{B}\) 个关键点,满足任意一个 ...

  2. 【BZOJ2589】 Spoj 10707 Count on a tree II

    BZOJ2589 Spoj 10707 Count on a tree II Solution 吐槽:这道题目简直...丧心病狂 如果没有强制在线不就是树上莫队入门题? 如果加了强制在线怎么做? 考虑 ...

  3. 【BZOJ2589】[SPOJ10707]Count on a tree II

    [BZOJ2589][SPOJ10707]Count on a tree II 题面 bzoj 题解 这题如果不强制在线就是一个很\(sb\)的莫队了,但是它强制在线啊\(qaq\) 所以我们就用到了 ...

  4. SPOJ COT2 - Count on a tree II(LCA+离散化+树上莫队)

    COT2 - Count on a tree II #tree You are given a tree with N nodes. The tree nodes are numbered from  ...

  5. COT2 - Count on a tree II(树上莫队)

    COT2 - Count on a tree II You are given a tree with N nodes. The tree nodes are numbered from 1 to N ...

  6. 【BZOJ2588】Count On a Tree(主席树)

    [BZOJ2588]Count On a Tree(主席树) 题面 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第 ...

  7. 【SPOJ10707】 COT2 Count on a tree II

    SPOJ10707 COT2 Count on a tree II Solution 我会强制在线版本! Solution戳这里 代码实现 #include<stdio.h> #inclu ...

  8. 【SPOJ】Count On A Tree II(树上莫队)

    [SPOJ]Count On A Tree II(树上莫队) 题面 洛谷 Vjudge 洛谷上有翻译啦 题解 如果不在树上就是一个很裸很裸的莫队 现在在树上,就是一个很裸很裸的树上莫队啦. #incl ...

  9. spoj COT2 - Count on a tree II

    COT2 - Count on a tree II http://www.spoj.com/problems/COT2/ #tree You are given a tree with N nodes ...

随机推荐

  1. HDFS Shell基本操作

    1.目录操作 hdfs dfs [命令]  [命令]         等价于            hadoop fs []  [] 1  ./bin/hdfs dfs -mkdir -p /user ...

  2. Spring中常用重要的接口

    Spring (ApplicationContext 初始化Bean的方法 refresh()) public void refresh() throws BeansException, Illega ...

  3. Linkerd 2.10(Step by Step)—控制平面调试端点

    Linkerd 2.10 系列 快速上手 Linkerd v2 Service Mesh(服务网格) 腾讯云 K8S 集群实战 Service Mesh-Linkerd2 & Traefik2 ...

  4. 写webpack插件报警告Tapable.plugin is deprecated. Use new API on .hooks instead解决方案,webpack4插件新写法

    最近写了个小插件报了个警告,然后去百度了一下,全都给我说extract-text-webpack-plugin这个插件有问题要更新,我也是无语了,这个插件我用都没用,百度翻了下齐刷刷全是这个答案,搞得 ...

  5. Win7/Win10+VS2017+OpenCV3.4.2安装、测试

    安装VS2017 在微软官网https://www.microsoft.com,下载Visual Studio 2017安装包 用管理员权限运行vs2017 enterprise安装包,安装过程会持续 ...

  6. MySQL大数据迁移备份

    MySQL迁移通常使用的有三种方法:   1.数据库直接导出,拷贝文件到新服务器,在新服务器上导入. 2.使用第三方迁移工具. 3.数据文件和库表结构文件直接拷贝到新服务器,挂载到同样配置的MySQL ...

  7. SpringBoot笔记(2)

    一.容器功能 1.1 组件添加 1. @Configuration Full模式:获取对象时,首先在容器内搜索是否存在,如存在直接拿出 默认为Full模式,单例 配置类组件之间有依赖关系,方法会被调用 ...

  8. Session原理、生命周期及购物车功能的实现

    在WEB开发中,服务器可以为每个用户浏览器创建一个会话对象(session对象),注意:一个浏览器独占一个session对象(默认情况下).因此,在需要保存用户数据(保存该浏览器(会话)的相关信息)时 ...

  9. 第18章-x86指令集之常用指令

    x86的指令集可分为以下4种: 通用指令 x87 FPU指令,浮点数运算的指令 SIMD指令,就是SSE指令 系统指令,写OS内核时使用的特殊指令 下面介绍一些通用的指令.指令由标识命令种类的助记符( ...

  10. java 使用匿名内部类实现多线程的创建

    匿名内部类的作用:简化代码 把子类继承父类,重写父类的方法,创建子类对象合一步完成 把实现类实现类接口,重写接口中的方法,创建实现类对象合成一步完成 匿名内部类的最终产物:子类/实现类对象,而这个类没 ...