【FJWC 2019】 森林

样例输入

0

5

1 0 0 2

样例输出

1

2

3

3

我们发现,答案就是直径加上直径上某个点出发,不经过其他直径上的点的最长链。这里的直径可以是任意一条直径。

首先我们每次只加一个点,所以我们很好维护新的直径。假设旧直径的两个端点是\((A,B)\),则加入点\(X\)后新的端点可能是\((A,B),(A,X),(B,X)\)。

然后我们考虑求“直径上某个点出发,不经过其他直径上的点的最长链”。

我们知道,\(Lct\)有虚边和实边。我们给每个节点开一个\(multiset\)维护虚子树贡献的最长链。

我们记录\(mx_v\)表示\(v\)所在\(splay\)中所有虚儿子贡献的最长链。\(lmx_v\)表示\(v\)所在\(splay\)的从最左端点出发,经过一段实边,再经过一段虚边的最长路径;\(rmx_v\)同理。

很显然,\(v\)所在这条实链的顶端到子树内的最长链就是\(lmx_v\)。记录\(lmx_v\)是为了在\(access\)操作的时候维护向其父亲贡献的最长链。

我们询问的时候就先\(MakeRoot(A)\),再\(access(B)\),这样\(A\to B\)的直径在一条实链上,答案就是\(dis_{A,B}+mx_A-[mx_A!=0]\)。

因为有\(MakeRoot\)操作,所以我们要维护\(rmx\),\(reverse\)的时候还要交换\(lmx,rmx\)。

可以参考【清华集训2016】数据交互

注意:\(push\_down(v)\)的时候要将左右儿子的\(lmx\)和\(rmx\)也交换了,否则\(update\)的时候会出错。

代码:

#include<bits/stdc++.h>
#define ll long long
#define N 400005 using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} multiset<int>st[N];
int n;
int ans;
int A,B;
int rev[N],fa[N],ch[N][2];
int lmx[N],rmx[N],mx[N];
int size[N];
#define ls ch[v][0]
#define rs ch[v][1] void update(int v) {
size[v]=size[ls]+size[rs]+1;
mx[v]=max(*(--st[v].end()),max(mx[ls],mx[rs]));
lmx[v]=*(--st[v].end())+size[ls];
rmx[v]=*(--st[v].end())+size[rs];
lmx[v]=max(lmx[v],lmx[ls]);
rmx[v]=max(rmx[v],rmx[rs]);
if(rs) lmx[v]=max(lmx[v],lmx[rs]+size[ls]+1);
if(ls) rmx[v]=max(rmx[v],rmx[ls]+size[rs]+1);
} void Rev(int v) {
rev[v]^=1;
swap(ls,rs);
swap(lmx[v],rmx[v]);
} void down(int v) {
if(rev[v]) {
/*************/
Rev(ls);
Rev(rs);
rev[v]=0;
}
} bool isroot(int v) {return v!=ch[fa[v]][0]&&v!=ch[fa[v]][1];} void rot(int v) {
int f=fa[v],gr=fa[f];
int sn=v==ch[f][1],son=ch[v][!sn];
if(!isroot(f)) ch[gr][f==ch[gr][1]]=v;
ch[f][sn]=son;
ch[v][!sn]=f;
if(son) fa[son]=f;
fa[v]=gr;
fa[f]=v;
update(f);
update(v);
} void Splay(int v) {
static int st[N],top;
top=0;
st[++top]=v;
for(int i=v;!isroot(i);i=fa[i]) st[++top]=fa[i];
while(top) down(st[top--]);
while(!isroot(v)) {
int f=fa[v],gr=fa[f];
if(!isroot(f)) rot(v==ch[f][1]^f==ch[gr][1]?v:f);
rot(v);
}
} void Insert(int v,int f) {st[f].insert(lmx[v]+1);}
void Del(int v,int f) {st[f].erase(st[f].find(lmx[v]+1));} void access(int v) {
int tem=0;
while(v) {
Splay(v);
if(tem) Del(tem,v);
if(rs) Insert(rs,v);
rs=tem;
update(v);
tem=v;
v=fa[v];
}
} void Make_root(int v) {
access(v);
Splay(v);
Rev(v);
} void Link(int v,int f) {
update(v);
access(f);
Splay(f);
fa[v]=f;
Insert(v,f);
update(f);
} namespace DIS {
int dep[N],fa[N][20];
int mxdis=0;
int lca(int a,int b) {
if(dep[a]<dep[b]) swap(a,b);
for(int i=18;i>=0;i--)
if(fa[a][i]&&dep[fa[a][i]]>=dep[b])
a=fa[a][i];
if(a==b) return a;
for(int i=18;i>=0;i--)
if(fa[a][i]!=fa[b][i])
a=fa[a][i],b=fa[b][i];
return fa[a][0];
}
int dis(int a,int b) {return dep[a]+dep[b]-2*dep[lca(a,b)];}
void Insert(int v,int f) {
fa[v][0]=f;
for(int i=1;i<=18;i++) fa[v][i]=fa[fa[v][i-1]][i-1];
dep[v]=dep[f]+1;
int da=dis(A,v),db=dis(B,v);
mxdis=max(mxdis,max(da,db));
if(da==mxdis) B=v;
else if(db==mxdis) A=v;
}
} int main() {
int cas=Get();
n=Get();
for(int i=1;i<=n;i++) st[i].insert(0);
A=B=1;
for(int i=2;i<=n;i++) {
int a=Get()^ans;
DIS::Insert(i,a);
Link(i,a);
Make_root(A);
access(B);
Splay(B);
cout<<(ans=DIS::mxdis+mx[B]-(mx[B]!=0))<<"\n";
}
return 0;
}

【FJWC 2019】 森林的更多相关文章

  1. 【FJWC 2019】min

    [FJWC 2019]min 题目描述 给你一张 \(n\) 个点 \(m\) 条边的无向图,走过每条边都需要花费 \(1\) 秒. 给你一个整数 \(k\) ,请你选择至多 \(k\) 个点,令经过 ...

  2. FJWC 2019 游记

    FJWC 2019 游记 Day 0 春节旅游, 刚从杭州绍兴一带赶回来, 然而并没有直接飞去福州, 去了厦门再去福州, 浪费了好多时间. Day 1 酒店到学校有 \(20\) 分钟的步行路程, 感 ...

  3. 【NOI2019模拟2019.7.1】为了部落 (生成森林计数,动态规划)

    Description: \(1<=n<=1e9,1<=m,k<=100\) 模数不是质数. 题解: 先选m个点,最后答案乘上\(C_{n}^m\). 不妨枚举m个点的度数和D ...

  4. 「ZJOI2016」大森林 解题报告

    「ZJOI2016」大森林 神仙题... 很显然线段树搞不了 考虑离线操作 我们只搞一颗树,从位置1一直往后移动,然后维护它的形态试试 显然操作0,1都可以拆成差分的形式,就是加入和删除 因为保证了操 ...

  5. 先森林后树木:Elasticsearch各版本升级核心内容必看

    在学习Elasticsearch 时候,因为各个版本的问题,搞不清,非常的头疼,官方也给出了各个版本更新的情况,不过是英文版本,版本更新信息又特别多,最近学习,看了很多资料,没有一个整理很清楚的,然后 ...

  6. 【主席树启发式合并】【P3302】[SDOI2013]森林

    Description 给定一个 \(n\) 个节点的森林,有 \(Q\) 次操作,每次要么将森林中某两点联通,保证操作后还是个森林,要么查询两点间权值第 \(k\) 小,保证两点联通.强制在线. L ...

  7. 2019/8/27 Test(luogu 五月天模拟赛)

    \(2019/8/27\)大考 \(\color{#ff0808}{\text{初二诀别赛(SAD)}}\) 题目名称 链接 寿司 \(BSOJ5111\) 秀秀的森林 \(BSOJ5125\) 分组 ...

  8. HNOI 2019 多边形

    HNOI 2019 多边形 题意 小 R 与小 W 在玩游戏. 他们有一个边数为\(n\)的凸多边形,其顶点沿逆时针方向标号依次为\(1,2,3...n\).最开始凸多边形中有\(n\)条线段,即多边 ...

  9. [白话解析] 通俗解析集成学习之bagging,boosting & 随机森林

    [白话解析] 通俗解析集成学习之bagging,boosting & 随机森林 0x00 摘要 本文将尽量使用通俗易懂的方式,尽可能不涉及数学公式,而是从整体的思路上来看,运用感性直觉的思考来 ...

随机推荐

  1. ___树形菜单Ztree.js显示.

    ----视图@{ Layout = null;} <!DOCTYPE html><HTML><HEAD> <TITLE> ZTREE DEMO - be ...

  2. tomcat域名绑定设置

    域名绑定分为单域名绑定.多域名绑定,配置主要涉及到tomcat目录下conf/server.xml文件 一.单域名绑定 1.修改server.xml 大约105行的内容(不是必须修改,如果只是绑定一个 ...

  3. mysql创建和调用out参数的存储过程

    CREATE PROCEDURE sp_add(a int, b int,out c int) begin set c=a+ b; end; 调用过程: call sp_add (,,@a); sel ...

  4. sudo: cd: command not found

    事件起因 今天在aws ubutun上忽然发现的一个问题,执行sudo cd 时出现 sudo: cd: command not found 原因 shell shell是一个命令解析器 所谓shel ...

  5. Base64字符保存图片,图片转换成Base64字符编码

    //文件转换成Base64编码 public static String getFileBase64Str(String filePath) throws IOException { String f ...

  6. 定时任务Crontab

    0.基本概念 & 实现原理  定时任务基本概念: 调度器:负责管理Quartz应用运行时环境,用于调度定时任务. 定时任务:按照某种时间规则,被调度的任务. a.从有无状态来说,有以下两种: ...

  7. springboot 数据验证

    不能相信前端传过来的任何数据 一定不能相信前端传过来的任何数据 绝对不能相信前端传过来的任何数据 @JsonFormat 时间必须是指定的格式(这里是接收参数格式,不是取数据来格式化) @Null 必 ...

  8. 简单选择排序算法的C++实现

    简单选择排序采用最简单的选择方法,即在剩余序列中选出最小(或最大)的关键字,和剩余序列的第一个关键字交换位置,依次选择下去,直至使整个序列有序. 算法中两层循环的执行次数和初始序列没有关系,第二层循环 ...

  9. MySQL 性能优化-数据库死锁监控

    MySQL性能优化-数据库死锁监控 by:授客 QQ:1033553122 1)表锁定 通过检查 table_locks_waited 和 table_locks_immediate 状态变量来分析表 ...

  10. Python基础知识点

    自学记录: 1.字符串 python中单引号和双引号使用完全相同. 使用三引号('''或""")可以指定一个多行字符串. 转义符 '\' 反斜杠可以用来转义,使用r可以让 ...