[BJOI2014]大融合
XI.[BJOI2014]大融合
终于来了……我们终于要用LCT来维护子树信息了。
因为我们看到,LCT是通过将原树拆成一堆链而起效的。在树链剖分中,我们通过dfs序来访问一棵子树;但是因为LCT的链是动态变化的,因此并没有一组固定的访问顺序。
那怎么办呢?
我们考虑最原始的想法:对于每个节点,再额外维护所有虚子节点的状态。比如说,本题我们维护的就是虚子节点的子树大小之和。
我们设 t[x].s1
表示一个节点所有子节点(不管虚实)的子树大小之和,即在原树中,它自己的子树大小。再设 t[x].s2
为所有虚子节点的大小之和。
那么,pushup
函数将会长成这样:
void pushup(int x){
t[x].s1=t[x].s2+t[lson].s1+t[rson].s1+1;
}
我们再考虑其它函数会有什么变化。显然,只有当节点间的虚实关系(边由实转序、连边、断边等)发生变化时,s2
才会发生变化。
splay
函数:在同一棵实splay中操作,没有虚实关系变化。
access
函数:有变化!
我们拎出来 access
函数:
void access(int x){
for(int y=0;x;x=t[y=x].fa)splay(x),rson=y,pushup(x);
}
可以看到,我们有 rson=y
,虚实关系产生了变化!!!
对于,这是实转虚,s2
应该加上的 s1
;对于,这是虚转实,s2
应该减去的 s1
。
因此我们最终有:
void access(int x){
for(int y=0;x;x=t[y=x].fa)splay(x),t[x].s2+=t[rson].s1-t[y].s1,rson=y,pushup(x);
}
makeroot
, findroot
, split
函数:并没有虚实变化(变化全在函数内调用的 access
函数,已经在 access
时改过了),没有变化。
link
:情况有变!我们这时必须要将的 s2
加上的 s1
,因为成为了的虚儿子!
但是,一变,的父亲也要跟着变,的爷爷也是……
我们有办法。直接将 access
后移到(注意是splay的,不是整棵树的),这样就没有父亲了!
因此我们的 link
函数就变成了这样:
void link(int x,int y){
split(x,y);
t[y].s2+=t[x].s1;
t[x].fa=y;
pushup(y);
}
等等,哪来的 split
?
偷懒一下,我们只是把 makeroot(x),access(y),splay(y)
三合一。实际上和并不连通。
至于 cut
,这题并不需要。不过代码还是放出来:
void cut(int x,int y){
split(x,y);
t[x].fa=t[y].ch[0]=0;
pushup(y);
}
然后,当我们要求出一个节点的子树大小时:
int query(int x,int y){
split(x,y);
return t[x].s1;
}
其中就是指定的的父亲(不然不知道是哪颗子树),在这题中就是给定询问的边的另一端点。
完整版放出:
#include<bits/stdc++.h>
using namespace std;
#define lson t[x].ch[0]
#define rson t[x].ch[1]
int n,m;
struct node{
int ch[2],s1,s2,fa;
bool rev;
}t[100100];
int identify(int x){
if(t[t[x].fa].ch[0]==x)return 0;
if(t[t[x].fa].ch[1]==x)return 1;
return -1;
}
void pushup(int x){
t[x].s1=t[x].s2+t[lson].s1+t[rson].s1+1;
}
void REV(int x){
t[x].rev^=1,swap(lson,rson);
}
void pushdown(int x){
if(!t[x].rev)return;
if(lson)REV(lson);
if(rson)REV(rson);
t[x].rev=0;
}
void rotate(int x){
int y=t[x].fa;
int z=t[y].fa;
int dirx=identify(x);
int diry=identify(y);
int b=t[x].ch[!dirx];
if(diry!=-1)t[z].ch[diry]=x;t[x].fa=z;
if(b)t[b].fa=y;t[y].ch[dirx]=b;
t[x].ch[!dirx]=y,t[y].fa=x;
pushup(y),pushup(x);
}
void pushall(int x){
if(identify(x)!=-1)pushall(t[x].fa);
pushdown(x);
}
void splay(int x){
pushall(x);
while(identify(x)!=-1){
int fa=t[x].fa;
if(identify(fa)==-1)rotate(x);
else if(identify(fa)==identify(x))rotate(fa),rotate(x);
else rotate(x),rotate(x);
}
}
void access(int x){
for(int y=0;x;x=t[y=x].fa)splay(x),t[x].s2+=t[rson].s1-t[y].s1,rson=y,pushup(x);
}
void makeroot(int x){
access(x),splay(x),REV(x);
}
int findroot(int x){
access(x),splay(x);
pushdown(x);
while(lson)x=lson,pushdown(x);
splay(x);
return x;
}
void split(int x,int y){
makeroot(x),access(y),splay(y);
}
void link(int x,int y){
split(x,y);
t[y].s2+=t[x].s1;
t[x].fa=y;
}
int query(int x,int y){
split(x,y);
return t[x].s1;
}
int main(){
scanf("%d%d",&n,&m);
for(int i=1,x,y;i<=m;i++){
char s[10];
scanf("%s%d%d",s,&x,&y);
if(s[0]=='A')link(x,y);
else printf("%lld\n",1ll*query(x,y)*query(y,x));
}
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
· 【杂谈】分布式事务——高大上的无用知识?