题目大意:给你一颗n个节点的树,最初点集S为空。

有m次操作:往当前点集S中加入/删除一个点,询问点x至集合S中任意点的最小距离,回到第t次修改点集的操作后的状态。

数据范围:$n,m≤10^5$

我们先无视这个可持久化的要求,考虑下不可持久化怎么做。

显然考虑动态树分治。

令点v为当前分治中心,u为v在点分树上的父亲,

每个点开一个数组D,D[x]表示以v为根的点分树中,与v距离为不大于x的点的标记点数量。

我们借助这个数组,可以方便地求出从v走最少多少步可以走到一个标记点。

首先考虑查询操作,我们可以直接在点分树上从x开始往上跳,设当前跳到了点y,我们需要在这个点的D数组中找到一个最大的k,满足D[k]=0。 那么我们不难发现此时找到的距离点x最近的点距离为$k+dis(x,y)+1$。

至于为什么不需要像正常动态点分治那样容斥相减,那是因为此处我们只需要找到一个距离最近的即可,不去重也不会对答案有影响(这是我后来才发现的,场上写了相减的。。。。)

至于修改操作,我们直接在点分树上从x开始网上跳,设当前跳到了y,我们修改下$D[dis(x,y)]$后面的数据即可。

我们为了优化复杂度显然不可以暴力修改/查询D数组,在这里我们用线段树维护D数组即可。

然而此题中还要求要可持久化,把线段树换成可持久化线段树就可以了。

时间复杂度:$O(n\log^2\ n)$

代码后来优化了一波,不算太长。

 #include<bits/stdc++.h>
#define M 100005
#define N 20000005
using namespace std; 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 T=,newT=,n,pointcnt[M]={}; int f[M][]={},dep[M]={};
void dfs(int x,int fa){
f[x][]=fa; dep[x]=dep[fa]+;
for(int i=;i<;i++) f[x][i]=f[f[x][i-]][i-];
for(int i=head[x];i;i=e[i].next) if(e[i].u!=fa) dfs(e[i].u,x);
}
int getlca(int x,int y){
if(dep[x]<dep[y]) swap(x,y); int cha=dep[x]-dep[y];
for(int i=;~i;i--) if((<<i)&cha) x=f[x][i];
for(int i=;~i;i--) if(f[x][i]!=f[y][i]) x=f[x][i],y=f[y][i];
if(x==y) return x; return f[x][];
}
int getdis(int x,int y){return dep[x]+dep[y]-*dep[getlca(x,y)];} int fa[M]={},vis[M]={},siz[M]={},minn=,minid=;
void dfssiz(int x,int fa){
siz[x]=;
for(int i=head[x];i;i=e[i].next) if(e[i].u!=fa&&vis[e[i].u]==) dfssiz(e[i].u,x),siz[x]+=siz[e[i].u];
}
void dfsmax(int x,int fa,int fsiz){
int maxn=fsiz-siz[x];
for(int i=head[x];i;i=e[i].next) if(e[i].u!=fa&&vis[e[i].u]==) dfsmax(e[i].u,x,fsiz),maxn=max(maxn,siz[e[i].u]);
if(maxn<minn) minn=maxn,minid=x;
}
int makeroot(int x){dfssiz(x,); minn=M; dfsmax(x,,siz[x]); return minid;}
void solve(int x,int Fa){
x=makeroot(x); vis[x]=; fa[x]=Fa;
for(int i=head[x];i;i=e[i].next) if(vis[e[i].u]==) solve(e[i].u,x);
} int lc[N]={},rc[N]={},sum[N]={},use=;
void updata(int &x,int l,int r,int k,int val){
use++; lc[use]=lc[x]; rc[use]=rc[x]; sum[use]=sum[x]+val;
x=use; int mid=(l+r)>>; if(l==r) return;
if(k<=mid) updata(lc[x],l,mid,k,val); else updata(rc[x],mid+,r,k,val);
}
int query(int x,int l,int r){
if(sum[x]==) return r;
if(l==r) return -; int mid=(l+r)>>;
int res=query(lc[x],l,mid);
if(res!=mid) return res;
int res2=query(rc[x],mid+,r);
if(res2!=-) return res2; return res;
} struct hh{
int rt; hh(int RT=){rt=RT;}
int query(int x,int l,int r,int k){
if(l==r) return sum[x]; int mid=(l+r)>>;
if(k<=mid) return query(lc[x],l,mid,k);
return query(rc[x],mid+,r,k);
}
int set(int &x,int l,int r,int k,int val){
use++; lc[use]=lc[x]; rc[use]=rc[x]; x=use;
if(l==r) return sum[x]=val;
int mid=(l+r)>>;
if(k<=mid) set(lc[x],l,mid,k,val);
else set(rc[x],mid+,r,k,val);
}
int Query(int id){return query(rt,,n,id);}
int Set(int id,int val){return set(rt,,n,id,val);}
}a[M],on[M]; void Updata(int x,int id,int val){
int rt=a[newT].Query(x);
updata(rt,,n,getdis(x,id),val);
a[newT].Set(x,rt);
if(fa[x]) Updata(fa[x],id,val);
}
int Query(int x,int y){
int res=query(a[T].Query(x),,n)+getdis(x,y);
if(fa[x]) res=min(res,Query(fa[x],y));
return res;
} int main(){
scanf("%d",&n);
for(int i=,x,y;i<n;i++) scanf("%d%d",&x,&y),add(x,y),add(y,x);
dfs(,);
solve(,);
int m,ans=; scanf("%d",&m);
while(m--){
int op,x; scanf("%d%d",&op,&x); x^=ans;
if(op==){T=x; continue;}
if(op==){
newT++; a[newT]=a[T]; on[newT]=on[T];
int sel=on[T].Query(x);
if(sel==){
pointcnt[newT]=pointcnt[T]+;
on[newT].Set(x,);
}else{
pointcnt[newT]=pointcnt[T]-;
on[newT].Set(x,);
}
Updata(x,x,sel==?:-);
T=newT;
}else{
if(pointcnt[T]==) {printf("%d\n",ans=1e9); continue;}
printf("%d\n",ans=Query(x,x)+);
}
}
}

【xsy2818】 最近点 动态树分治+可持久化线段树的更多相关文章

  1. 主席树(可持久化线段树) 静态第k大

    可持久化数据结构介绍 可持久化数据结构是保存数据结构修改的每一个历史版本,新版本与旧版本相比,修改了某个区域,但是大多数的区域是没有改变的, 所以可以将新版本相对于旧版本未修改的区域指向旧版本的该区域 ...

  2. [HNOI2016]树(可持久化线段树+树上倍增)

    [HNOI2016]树(可持久化线段树+树上倍增) 题面 给出一棵n个点的模板树和大树,根为1,初始的时候大树和模板树相同.接下来操作m次,每次从模板树里取出一棵子树,把它作为新树里节点y的儿子.操作 ...

  3. hdu2665 主席树(可持久化线段树)

    题意:给定一个数组,每次查询第l到r区间的第k大值 解法嘛,当然是主席树,主席树即可持久化线段树,什么叫可持久化呢,就是指能够访问历史版本的数据结构,那么对于某些只能离线处理的题目强制在线之后 ,可以 ...

  4. cdqz2017-test10-rehearsal(CDQ分治&可持久化线段树&单调栈)

    题意: 给出n个三元组 e[i]=(si,ti,wi) 第i个三元组的价值为 Σ w[j] ,j 满足以下4个条件: 1.j<i 2.tj<ti 3.sj<si 4.不存在j< ...

  5. bzoj 4137 [FJOI2015]火星商店问题——线段树分治+可持久化01trie树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4137 关于可持久化01trie树:https://www.cnblogs.com/LadyL ...

  6. [FJOI2015]火星商店问题(线段树分治,可持久化,Trie树)

    [FJOI2015]火星商店问题 前天考了到线段树分治模板题,全场都切了,就我不会QAQ 于是切题无数的Tyher巨巨就告诉我:"你可以去看看火星商店问题,看了你就会了." 第一道 ...

  7. 线段树分治总结(线段树分治,线段树,并查集,树的dfn序,二分图染色)

    闲话 stO猫锟学长,满脑子神仙DS 网上有不少Dalao把线段树分治也归入CDQ分治? 还是听听YCB巨佬的介绍: 狭义:只计算左边对右边的贡献. 广义:只计算外部对内部的贡献. 看来可以理解为广义 ...

  8. BZOJ 4771 七彩树(可持久化线段树合并)

    题意 https://www.lydsy.com/JudgeOnline/problem.php?id=4771 思路 和 HDU 3333 其实有点像,不过是把序列的问题放在了树上,多维护一个深度即 ...

  9. 主席树(可持久化线段树)静态区间第K小

    传送门主席树 #include <bits/stdc++.h> #define int long long using namespace std; const int maxn=2e5+ ...

随机推荐

  1. ACM-ICPC 2018 徐州赛区网络预赛 J Maze Designer(最大生成树,倍增lca)

    https://nanti.jisuanke.com/t/31462 要求在一个矩形中任意选两个点都有唯一的通路,所以不会建多余的墙. 要求满足上述情况下,建墙的费用最小.理解题意后容易想到首先假设全 ...

  2. Windows下python环境配置

    步骤: 1.安装Python.Sublime Text: 2.打开Sublime Text,在菜单栏点击“Tools”->“Build System”->“New Build System ...

  3. s4-2 ALOHA 协议

    多路访问协议  随机访问协议(Random Access) 特点:站点争用信道,可能出现站点之间的冲突 典型的随机访问协议 • ALOHA协议 • CSMA协议 • CSMA/CD协议(以太网采 ...

  4. 1114 Family Property

    This time, you are supposed to help us collect the data for family-owned property. Given each person ...

  5. 2017-11-29 由runnable说起Android中的子线程和主线程

    1.首先纠正一个观点,就是runnable运行在子线程中是错误的观念.runnable只是创建了一个执行任务的对象,但是它本身并不会创建一个新的子线程,Runable只是给你接口让你实现工作线程的工作 ...

  6. android-基础编程-ListView

    ListView主要包括view和数据源.其数据适配器列表分为三种,ArrayAdapter,SimpleAdapter和SimpleCursorAdapter. ListView的没有oom原因.经 ...

  7. Lombok自定义annotation扩展含Intellij插件

    Lombok简介 Lombok(https://projectlombok.org/)  提供了以注解的形式为java对象增加属性和方法,这使得原来冗长的java源文件变的简洁(不需要再使用ide去生 ...

  8. vc6中向vs2010迁移的几个问题

    vc6版本支持的库编译:CJ60lib 1. 用vs2010打开CJ60库的源码的dsw,强制打开 (1)设置项目属性的语言 因为,如果代码字符的编码集不一样,则会出现函数冲定义,参数冲突等问题,这可 ...

  9. [置顶] 获取网络数据中的数组显示成ListView的简单流程

    首先说一下  这是我自己的个人笔记,如果想看看,不用看细节,可以看流程. 定义一个线程池 ExecutorService pool = Executors.newFixedThreadPool(15) ...

  10. Python自动化开发 - 面向对象(一)

    本节内容 1.编程范式 面向过程编程 面向对象编程 2.面向对象编程介绍 类的语法 类与实例内存分配 构造方法 自定义方法 3.面向对象特性 一.编程范式 编程是程序员 用特定的语法+数据结构+算法组 ...