【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. oracle expdp自动备份脚本

    windows: @echo off echo ================================================ echo Windows环境下Oracle数据库的自动 ...

  2. [android] 保存文件到手机内存

    /*****************2016年5月4日 更新*******************************/ 知乎:Android 没有沙盒保护机制吗,WhatsApp 信息为何可被随 ...

  3. 工作笔记-table问题汇总(vue单文件组件)

    1.vue: computed里定义的数据,在其他地方不能再重新赋值,会报错: Computed property "xxxxxx" was assigned to but it ...

  4. windows for windows:下载、安装

    关于docker的下载:同学们不要再网上找来找去了,直接去到阿里云下载客户端,配置阿里的国内镜像地址就好~ 申请一个阿里云的账号,然后: tips:如果使用虚拟机的os 是windows 那么很抱歉, ...

  5. vuejs 1.x - 实例:搜索过滤

    <!DOCTYPE html> <html> <head> <meta name="viewport" content="wid ...

  6. POJ 1704 Georgia and Bob(阶梯Nim博弈)

    Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 11357   Accepted: 3749 Description Geor ...

  7. dpr,ppi,dip,viewport的一些概念

    一 ppi,dpr,dip,分辨率,屏幕尺寸,设备物理像素,设备独立像素 分辨率:1920px*1080px 在这个设备上纵向上有 1920个像素点(色块)... 屏幕尺寸:5英寸(in) = 5*2 ...

  8. Angular调用Asp.net Core JWT Authentication接口

    基本思路是调用登录接口,获取token,使用token请求其他JWT接口: getHomeDetails(): Observable<HomeDetails> { let headers ...

  9. chrome正确的打开方式

    1:修改默认的搜索引擎 原因是中国不能使用Google浏览器,所以需要对其默认的搜索引擎进行改造:   三个点/设置/修改默认搜索引擎     2:使用插件;   右上角的省略号小点/更多工具/扩展应 ...

  10. 项目初始化mysql建库和授权

    创建数据库和授权 mysql -e "create database DATABASE DEFAULT CHARACTER SET utf8 DEFAULT COLLATE utf8_gen ...