BZOJ4573:[ZJOI2016]大森林——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=4573
https://www.luogu.org/problemnew/show/P3348#sub
小Y家里有一个大森林,里面有n棵树,编号从1到n。一开始这些树都只是树苗,只有一个节点,标号为1。这些树都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力。
小Y掌握了一种魔法,能让第l棵树到第r棵树的生长节点长出一个子节点。同时她还能修改第l棵树到第r棵树的生长节点。她告诉了你她使用魔法的记录,你能不能管理她家的森林,并且回答她的询问呢?
参考:http://www.sigongzi.org/index.php/archives/LOJ2092.html
思路:
显然我们不能对每棵树LCT维护一下,而且我们对于这棵树的形态还很严格。
那么我们把前一棵树的形态转换为后一棵树的形态,这样就只需要一棵树了。
先考虑对于0操作,实际上我们可以记录每个时刻每个节点在哪一段区间中(代码的L和R就是干这个的),所以我们大可以对所有的树都进行0操作。
对于1操作,和0操作类似,用L和R更新l和r后进行操作。
然后为了能够快捷的更新树,我们建立一个size为0的虚点(这样对于路径长度就不需要修改了),所有的生长操作都在上面进行,这样我们删除的时候cut这个点即可。
对于2操作,事实上两个点一定存在的话,完全可以让0和1操作全部排到它的前面。
实现:
先把所有操作存下来,然后以操作的树为第一关键字,操作编号和顺序为第二关键字排序。
(对于区间修改思考差分,毕竟我们都是对同一棵树操作的。)
然后按树编号从左到右进行操作,对于0和1操作先对虚点清空然后长即可。
查询的时候就是用LCA求最短路的方法一样。
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cctype>
#include<cstdio>
#include<vector>
#include<queue>
#include<cmath>
using namespace std;
const int N=4e5+;
struct data{
int pos,id,from,to;
}qry[N];
int n,m,fa[N],tr[N][],val[N],size[N];
int cnt,tot,sum,aux,L[N],R[N],id[N],ans[N];
inline bool cmp(data a,data b){
return a.pos<b.pos||(a.pos==b.pos&&a.id<b.id);
}
inline int read(){
int X=,w=;char ch=;
while(!isdigit(ch)){w|=ch=='-';ch=getchar();}
while(isdigit(ch))X=(X<<)+(X<<)+(ch^),ch=getchar();
return w?-X:X;
}
inline bool get(int x){
return tr[fa[x]][]==x;
}
inline bool isroot(int x){
if(!fa[x])return ;
return tr[fa[x]][]!=x&&tr[fa[x]][]!=x;
}
inline void upt(int x){
size[x]=val[x];
if(tr[x][])size[x]+=size[tr[x][]];
if(tr[x][])size[x]+=size[tr[x][]];
}
inline void rotate(int x){
int y=fa[x],z=fa[y],b=tr[y][]==x?tr[x][]:tr[x][];
if(z&&!isroot(y))(tr[z][]==y?tr[z][]:tr[z][])=x;
fa[x]=z;fa[y]=x;b?fa[b]=y:;
if(tr[y][]==x)tr[x][]=y,tr[y][]=b;
else tr[x][]=y,tr[y][]=b;
upt(y);upt(x);
}
inline void splay(int x){
while(!isroot(x)){
if(!isroot(fa[x]))
rotate((get(x)==get(fa[x])?fa[x]:x));
rotate(x);
}
upt(x);
}
inline int access(int x){
int y;
for(y=;x;y=x,x=fa[x]){
splay(x);tr[x][]=y;
if(y)fa[y]=x;
}
return y;
}
inline void link(int x,int y){
splay(y);fa[x]=y;
}
inline void cut(int x){
access(x);splay(x);
if(tr[x][])fa[tr[x][]]=;
tr[x][]=;upt(x);
}
inline void makenode(int x){
val[++sum]=x;size[sum]=x;
}
int main(){
n=read(),m=read();
cnt=;L[cnt]=,R[cnt]=n;id[cnt]=;
makenode();makenode();
aux=sum;
link(aux,id[]);
for(int i=;i<=m;i++){
int op=read();
if(op==){
L[++cnt]=read(),R[cnt]=read();
makenode();
id[cnt]=sum;
qry[++tot]=(data){,i-m,sum,aux};
}
if(op==){
int l=read(),r=read(),x=read();
l=max(l,L[x]);r=min(r,R[x]);
if(l<=r){
makenode();
link(sum,aux);
qry[++tot]=(data){l,i-m,sum,id[x]};
qry[++tot]=(data){r+,i-m,sum,aux};
aux=sum;
}
}
if(op==){
int l=read(),u=read(),v=read();
qry[++tot]=(data){l,i,id[u],id[v]};
}
}
memset(ans,-,sizeof(ans));
sort(qry+,qry+tot+,cmp);
for(int i=,p=;i<=tot;p++){
while(qry[i].pos==p){
if(qry[i].id>){
ans[qry[i].id]=;
access(qry[i].from);splay(qry[i].from);
ans[qry[i].id]+=size[qry[i].from];
int lca=access(qry[i].to);
splay(qry[i].to);
ans[qry[i].id]+=size[qry[i].to];
access(lca);splay(lca);ans[qry[i].id]-=size[lca]*;
}
else{
cut(qry[i].from);link(qry[i].from,qry[i].to);
}
i++;
}
}
for(int i=;i<=m;i++){
if(ans[i]>=)printf("%d\n",ans[i]);
}
return ;
}
+++++++++++++++++++++++++++++++++++++++++++
+本文作者:luyouqi233。 +
+欢迎访问我的博客:http://www.cnblogs.com/luyouqi233/+
+++++++++++++++++++++++++++++++++++++++++++
BZOJ4573:[ZJOI2016]大森林——题解的更多相关文章
- BZOJ4573 : [Zjoi2016]大森林
扫描线,从左到右依次处理每棵树. 用set按时间顺序维护影响了这棵树的所有操作,那么一个点的父亲就是它前面第一个操作1. 用Splay维护树的括号序列,那么两点间的距离就是括号数量减去匹配的括号个数. ...
- [BZOJ4573][ZJOI2016]大♂森林
bzoj luogu uoj sol \(orz\ \ HJT\ \ dalao\)教会我做这道题. 考虑每两个相邻位置的树的差异. 对于一个1操作(更换生长节点),假设区间是\([l,r]\),那么 ...
- [ZJOI2016]大森林(LCT)
题目描述 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力. 小Y掌握了一种 ...
- bzoj 4573: [Zjoi2016]大森林
Description 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树 都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力. ...
- P3348 [ZJOI2016]大森林
\(\color{#0066ff}{ 题目描述 }\) 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点, ...
- [ZJOI2016]大森林
Description: 小Y家里有一个大森林,里面有n棵树,编号从1到n 0 l r 表示将第 l 棵树到第 r 棵树的生长节点下面长出一个子节点,子节点的标号为上一个 0 号操作叶子标号加 1(例 ...
- 【刷题】BZOJ 4573 [Zjoi2016]大森林
Description 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力.小 ...
- 【LuoguP3348】[ZJOI2016]大森林
题目链接 题目描述 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力. 小Y ...
- 【BZOJ4573】[ZJOI2016] 大森林(LCT)
点此看题面 大致题意: 有\(n\)棵树,初始各有\(1\)个编号为\(1\)的节点,且其为生长节点.\(3\)种操作:将\([l,r]\)区间内的树增加一个新的编号的节点,修改\([l,r]\)区间 ...
随机推荐
- 你想找的Python资料这里全都有!没有你找不到!史上最全资料合集
你想找的Python资料这里全都有!没有你找不到!史上最全资料合集 2017年11月15日 13:48:53 技术小百科 阅读数:1931 GitHub 上有一个 Awesome - XXX 系列 ...
- Python 列表下标操作
Python 列表下标操作 引用网址: https://www.jianshu.com/p/a98e935e4d46
- React中类定义组件constructor 和super
刚开始学习React没多久,在老师的教程里看到了类组件的使用示例,但是和资料上有些冲突,而引发了一些疑问: 类组件中到底要不要定义构造函数constructor()? super()里边到底要不要传入 ...
- 「日常训练」Skills(Codeforce Round #339 Div.2 D)
题意(CodeForces 614D) 每个人有\(n(n\le 10^5)\)个技能,技能等级都在\([0,10^9]\)的范围,每个技能有一个当前等级,所有技能的最高等级都为A.一个人的力量被记做 ...
- spark写入ES(动态模板)
使用es-hadoop插件,主要使用elasticsearch-spark-20_2.11-6.2.x.jar 官网:https://www.elastic.co/guide/en/elasticse ...
- Paper Reading - Learning to Evaluate Image Captioning ( CVPR 2018 ) ★
Link of the Paper: https://arxiv.org/abs/1806.06422 Innovations: The authors propose a novel learnin ...
- POJ 2187 Beauty Contest(凸包+旋转卡壳)
Description Bessie, Farmer John's prize cow, has just won first place in a bovine beauty contest, ea ...
- VUE中组件的使用
关于vue组件引用 使用Nodejs的方法 被引用的组件要暴露 module.exports={}; 引用时 用 var abc= require("组件的路径") 然后 就可以用 ...
- Thunder团队第二周 - Scrum会议2
Scrum会议2 小组名称:Thunder 项目名称:爱阅app Scrum Master:胡佑蓉 工作照片: 参会成员: 王航:http://www.cnblogs.com/wangh013/ 李传 ...
- javabean的内省技术和BeanUtils的使用
一.关于javabean javabean是固定写法的java类 书写格式为: 1)必须有无参构造函数 2)属性必须私有, 我们称为字段 3)提供标准的getter和setter 例: name 字段 ...