bzoj

luogu

uoj

sol

\(orz\ \ HJT\ \ dalao\)教会我做这道题。

考虑每两个相邻位置的树的差异。

对于一个1操作(更换生长节点),假设区间是\([l,r]\),那么第\(l-1\)棵树和第\(l\)棵数就会产生一定的差异,具体来说,假设在这之后没有1操作了,那么所有应该在第\(l-1\)棵树中接到原生长节点下面的点都在第\(l\)棵树中被接到了新的生长节点的下面。

我们考虑怎么快速实现这个操作。很显然,我们只要把修改了生长节点后新长出来的点打个包,然后\(cut\)再\(link\)一下就好了。

如果是一棵子树,直接对根节点进行操作就行了。

但是显然并不是一棵子树啊。

那就直接新建一个点把新建的点接在下面就好了。

具体来说,对于每一个1操作,新建一个虚点,把所有从这个生长节点长出来的点接在下面。

在处理到第\(l\)棵树时,把这个虚点接到那个生长节点的下面去;处理到第\(r+1\)棵树的时候再把虚点接到前一个生长节点对应的虚点上去。

这样就可以直接查询两个点的路径长度了。

但是有新建的虚点?

直接把实点的权值设为1,虚点的权值设为0,然后查路径和就好了。

但是这是一棵有根树,并不能\(split\)啊。

所以可以查\(s[x]+s[y]-2*s[lca(x,y)]\)?

现在的问题是:怎么在\(LCT\)查\(LCA\)?

先\(access(x)\),然后在\(access(y)\)的时候,最后一个由虚边改为实边的位置就是\(LCA\)。

这个画个图就能理解了吧。

然后这题就做完了。

code

#include<cstdio>
#include<algorithm>
using namespace std;
int gi()
{
int x=0,w=1;char ch=getchar();
while ((ch<'0'||ch>'9')&&ch!='-') ch=getchar();
if (ch=='-') w=0,ch=getchar();
while (ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
return w?x:-x;
}
const int N = 4e5+5;
struct node{
int pos,tim,x,y;
bool operator < (const node &b) const
{return pos!=b.pos?pos<b.pos:tim<b.tim;}
}q[N];
int n,m,cnt,Q,fa[N],ch[2][N],val[N],sum[N],grow,real,tot,id[N],L[N],R[N],ans[N];
bool son(int x){return x==ch[1][fa[x]];}
bool isroot(int x)
{
return x!=ch[0][fa[x]]&&x!=ch[1][fa[x]];
}
void pushup(int x)
{
sum[x]=sum[ch[0][x]]+sum[ch[1][x]]+val[x];
}
void rotate(int x)
{
int y=fa[x],z=fa[y],c=son(x);
ch[c][y]=ch[c^1][x];if (ch[c][y]) fa[ch[c][y]]=y;
fa[x]=z;if (!isroot(y)) ch[son(y)][z]=x;
ch[c^1][x]=y;fa[y]=x;pushup(y);
}
void splay(int x)
{
for (int y=fa[x];!isroot(x);rotate(x),y=fa[x])
if (!isroot(y)) son(x)^son(y)?rotate(x):rotate(y);
pushup(x);
}
int access(int x)
{
int y=0;
for (;x;y=x,x=fa[x])
splay(x),ch[1][x]=y,pushup(x);
return y;
}
void link(int x,int y)
{
access(x);splay(x);fa[x]=y;
}
void cut(int x)
{
access(x);splay(x);
ch[0][x]=fa[ch[0][x]]=0;pushup(x);
}
int main()
{
// freopen("a.in","r",stdin);
// freopen("a.out","w",stdout);
n=gi();m=gi();link(2,1);val[1]=1;id[1]=1;
L[1]=1;R[1]=n;grow=tot=2;real=1;
for (int i=1;i<=m;++i)
{
int opt=gi(),l=gi(),r=gi();
if (!opt)
{
link(id[++real]=++tot,grow);val[tot]=1;
L[real]=l;R[real]=r;
}
if (opt==1)
{
int x=gi();l=max(l,L[x]),r=min(r,R[x]);
if (l>r) continue;
link(++tot,grow);
q[++cnt]=(node){l,i-m,tot,id[x]};q[++cnt]=(node){r+1,i-m,tot,grow};
grow=tot;
}
if (opt==2)
{
int x=gi();
q[++cnt]=(node){l,++Q,id[r],id[x]};
}
}
sort(q+1,q+cnt+1);
for (int i=1;i<=cnt;++i)
if (q[i].tim>0)
{
int res,gg;
access(q[i].x);splay(q[i].x);res=sum[q[i].x];
gg=access(q[i].y);splay(q[i].y);res+=sum[q[i].y];
access(gg);splay(gg);res-=sum[gg]<<1;
ans[q[i].tim]=res;
}
else cut(q[i].x),link(q[i].x,q[i].y);
for (int i=1;i<=Q;++i) printf("%d\n",ans[i]);
return 0;
}

[BZOJ4573][ZJOI2016]大♂森林的更多相关文章

  1. BZOJ4573 : [Zjoi2016]大森林

    扫描线,从左到右依次处理每棵树. 用set按时间顺序维护影响了这棵树的所有操作,那么一个点的父亲就是它前面第一个操作1. 用Splay维护树的括号序列,那么两点间的距离就是括号数量减去匹配的括号个数. ...

  2. BZOJ4573:[ZJOI2016]大森林——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=4573 https://www.luogu.org/problemnew/show/P3348#sub ...

  3. [ZJOI2016]大森林(LCT)

    题目描述 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力. 小Y掌握了一种 ...

  4. [ZJOI2016]大森林

    Description: 小Y家里有一个大森林,里面有n棵树,编号从1到n 0 l r 表示将第 l 棵树到第 r 棵树的生长节点下面长出一个子节点,子节点的标号为上一个 0 号操作叶子标号加 1(例 ...

  5. 【刷题】BZOJ 4573 [Zjoi2016]大森林

    Description 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力.小 ...

  6. bzoj 4573: [Zjoi2016]大森林

    Description 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树 都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力. ...

  7. P3348 [ZJOI2016]大森林

    \(\color{#0066ff}{ 题目描述 }\) 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点, ...

  8. 【LuoguP3348】[ZJOI2016]大森林

    题目链接 题目描述 小Y家里有一个大森林,里面有n棵树,编号从1到n.一开始这些树都只是树苗,只有一个节点,标号为1.这些树都有一个特殊的节点,我们称之为生长节点,这些节点有生长出子节点的能力. 小Y ...

  9. 【BZOJ4573】[ZJOI2016] 大森林(LCT)

    点此看题面 大致题意: 有\(n\)棵树,初始各有\(1\)个编号为\(1\)的节点,且其为生长节点.\(3\)种操作:将\([l,r]\)区间内的树增加一个新的编号的节点,修改\([l,r]\)区间 ...

随机推荐

  1. 流量分析系统----讲解-echarts模拟迁移(结合china.js)

    百度 Echarts 地图->模拟迁徙,实现自动切换地图 小航哥注释: 1.本文主要是把模拟迁移的流程讲了一遍,讲的很好.具体实现参考航哥这篇随笔“流量分析系统----实现-echarts模拟迁 ...

  2. POJ 3468 A Simple Problem with Integers 【线段树】

    题目链接 http://poj.org/problem?id=3468 思路 线段树 区间更新 模板题 在赋初始值的时候,按点更新区间就可以 AC代码 #include <cstdio> ...

  3. iOS 点击注释图标 弹出对应解释

    需求:如题目  接上一篇的开发内容 效果图: 这种情况存在tableView 的一个cell中. 要点 1,  弹出的对应解释 要在可视区域,并且小尖角 要指着 图片 2,  文本不能过高 有极大高度 ...

  4. HBase基本知识介绍及典型案例分析

    本次分享的内容主要分为以下五点: HBase基本知识: HBase读写流程: RowKey设计要点: HBase生态介绍: HBase典型案例分析. 首先我们简单介绍一下 HBase 是什么. HBa ...

  5. win32调试——OutputDebugString

    win32下开发console程序可以直接用printf打印到控制台. 开发图形界面程序时,可以调用OutputDebugString将字符串输出到Debug窗口, 注意是要调试运行才能看到Debug ...

  6. linux免密登录配置

    第一步:安装openssh-clients yum install -y openssh-clients.x86_64第二步:生成密钥 ssh-keygen第三步:拷贝公钥到其他机器 ssh-copy ...

  7. 【HackerRank】Service Lane

     Calvin is driving his favorite vehicle on the 101 freeway. He notices that the check engine light o ...

  8. for循环执行流程

    语句格式: for(表达式1;表达式2;表达式3) { 循环体 } 表达式1:赋值表达式,用来给控制变量赋初值.(只执行一次) 表达式2:逻辑表达式,是循环的控制条件,用来判断控制变量是否符合循环条件 ...

  9. 用nodejs实现读取文件操作

    //如果不是全局就得引入fs成员 const fs = require("fs"); //fs 核心模块中提供了一个 fs.readFile方法,来读取指定目录下的文件 //fs. ...

  10. C#多线程学习之Thread

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...