题面

给你一棵

n

n

n 个结点的树,对于所有的

k

[

0

,

n

]

k\in[0,n]

k∈[0,n] ,求出

M

E

X

=

k

{\rm MEX}=k

MEX=k 的路径数量。

一条路径的

M

E

X

\rm MEX

MEX 定义为该路径上没出现的最小结点编号(编号从

0

0

0 到

n

1

n-1

n−1)。

1

n

2

e

5

1\leq n\leq 2e5

1≤n≤2e5,多组数据。

题解

一条路径的

M

E

X

\rm MEX

MEX 恰好等于

k

k

k 的充要条件是

  • 结点

    0

    0

    0 到

    k

    1

    k-1

    k−1 都出现在路径上。

  • 结点

    k

    k

    k 没有出现在路径上。

我们先想想怎么求

0

0

0 到

i

i

i 都出现在路径上的路径个数,令其为

f

(

i

)

f(i)

f(i)。

如果这

i

+

1

i+1

i+1 个点在树上没办法由一条路径经过(形成的虚树不是一条链)那么答案一定为 0,并且

f

(

i

+

1

)

f(i+1)

f(i+1) 直到

f

(

n

)

f(n)

f(n) 的值也都是 0 。在这之前,如果包含它们的最短路径两端是

A

A

A 和

B

B

B(不妨令它们没有祖先关系),那么

f

(

i

)

f(i)

f(i) 就为

A

A

A 端子树大小 ×

B

B

B 端子树大小。

M

E

X

\rm MEX

MEX 等于

k

k

k 的答案不难发现可以用一个小容斥算出:

f

(

k

1

)

f

(

k

)

f(k-1)-f(k)

f(k−1)−f(k) 。

然后,难点就剩计算

f

(

i

)

f(i)

f(i) 了。首先不妨令

f

(

1

)

=

n

(

n

1

)

2

f(-1)=\cfrac{n*(n-1)}{2}

f(−1)=2n∗(n−1)​

接着我们发现,如果把 0 当作根的话,一切都变得明朗。从

0

0

0 到

i

i

i 形成的虚树一定都经过根,那么我们用

A

A

A 和

B

B

B 两个变量存路径的两端就足够了(初始为 0)。从小到大依次加入点,如果出现加入的点

x

x

x 分别与

A

A

A

B

B

B 的最近公共祖先不是

1

,

A

,

B

,

x

1,A,B,x

1,A,B,x 中任何一个的话,那当前的

f

(

i

)

f(i)

f(i) 以及后面的都是 0 了,就不用继续算了。否则的话,就讨论讨论,看怎么更新

A

A

A 和

B

B

B 。

计算

f

(

i

)

f(i)

f(i) 的时候,如果

A

A

A 和

B

B

B 都是 0,那么就是经过 0 的路径数,提前处理一下即可;如果

A

A

A 和

B

B

B 其中一个是 0(假设是

A

A

A),那么先求出

B

B

B 往上走刚好走到 0 的儿子的那个点

B

B'

B′,答案就是

(

n

s

i

z

[

B

]

)

×

s

i

z

[

B

]

(n-siz[B'])\times siz[B]

(n−siz[B′])×siz[B];如果

A

A

A 和

B

B

B 都大于 0 的话,答案自然是

s

i

z

[

A

]

×

s

i

z

[

B

]

siz[A]\times siz[B]

siz[A]×siz[B]。

具体实现有点细节多。复杂度

O

(

n

log

n

)

O(n\log n)

O(nlogn)。

CODE

#include<cstdio>
#include<vector>
#include<cmath>
#include<map>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 300005
#define DB double
#define LL long long
#define ENDL putchar('\n')
#define lowbit(x) ((-x) & (x))
LL read() {
LL f=1,x=0;char s = getchar();
while(s < '0' || s > '9') {if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s-'0');s = getchar();}
return f * x;
}
int n,m,i,j,s,o,k;
LL as[MAXN];
vector<int> g[MAXN];
int d[MAXN],dfn[MAXN],rr[MAXN],tim,siz[MAXN];
LL dp[MAXN];
int A,B,FLAG,f[MAXN][21];//18
void dfs(int x,int ff) {
d[x] = d[f[x][0] = ff] + 1;dp[x] = 0;siz[x] = 1;
dfn[x] = ++ tim;
for(int i = 1;i <= 18;i ++) f[x][i] = f[f[x][i-1]][i-1];
for(int i = 0;i < (int)g[x].size();i ++) {
int y = g[x][i];
if(y != ff) {
dfs(y,x);
dp[x] += siz[x] *1ll* siz[y];
siz[x] += siz[y];
}
}rr[x] = tim;
return ;
}
int lca(int a,int b) {
if(d[a] < d[b]) swap(a,b);
if(d[a] > d[b]) {
for(int i = 18;i >= 0;i --) {
if(d[f[a][i]] >= d[b]) a = f[a][i];
}
}if(a == b) return a;
for(int i = 18;i >= 0;i --) {
if(f[a][i] != f[b][i]) {
a = f[a][i]; b = f[b][i];
}
}return f[a][0];
}
LL cal() {
if(!FLAG) return 0;
if(A > B) swap(A,B);
if(A == 1 && B == 1) return dp[1];
if(A == 1) {
int ft = B;
for(int i = 18;i >= 0;i --) {
if(d[f[ft][i]] > d[A]) ft = f[ft][i];
}
int nm1 = n - (rr[ft]-dfn[ft]+1);
return (rr[B]-dfn[B]+1) *1ll* nm1;
}
return (rr[B]-dfn[B]+1) *1ll* (rr[A]-dfn[A]+1);
}
void addp(int x) {
if(!FLAG) return ;
if(A > B) swap(A,B);
if(A == 1 && B == 1) B = x;
else if(A == 1) {
int lc = lca(x,B);
if(lc == 1) A = x;
else if(lc == B) B = x;
else if(lc != x) FLAG = 0;
}
else {
int lc1 = lca(A,x),lc2 = lca(B,x);
if(lc1 == A) A = x;
else if(lc2 == B) B = x;
else if((lc1 == x) || (lc2 == x)) return ;
else FLAG = 0;
}return ;
}
int main() {
int T = read();
while(T --) {
n = read();
tim = 0;A = B = FLAG = 1;
for(int i = 1;i <= n+1;i ++) {
g[i].clear(); as[i] = 0;
memset(f[i],0,sizeof(f[i]));
}
for(int i = 1;i < n;i ++) {
s = read()+1;o = read()+1;
g[s].push_back(o);
g[o].push_back(s);
}
dfs(1,0);
as[1] = n*1ll*(n-1)/2ll;
for(int i = 2;i <= n;i ++) {
LL ass = cal();
as[i-1] -= ass;
as[i] = ass;
addp(i);
}
LL ass = cal();
as[n] -= ass; as[n+1] = ass;
for(int i = 1;i <= n+1;i ++) {
printf("%lld ",as[i]);
}ENDL;
}
return 0;
}

[CF1527D] MEX Tree (lca)的更多相关文章

  1. 面试题6:二叉树最近公共节点(LCA)《leetcode236》

    Lowest Common Ancestor of a Binary Tree(二叉树的最近公共父亲节点) Given a binary tree, find the lowest common an ...

  2. 洛谷P3379 【模板】最近公共祖先(LCA)

    P3379 [模板]最近公共祖先(LCA) 152通过 532提交 题目提供者HansBug 标签 难度普及+/提高 提交  讨论  题解 最新讨论 为什么还是超时.... 倍增怎么70!!题解好像有 ...

  3. Device Tree(三):代码分析【转】

    转自:http://www.wowotech.net/linux_kenrel/dt-code-analysis.html Device Tree(三):代码分析 作者:linuxer 发布于:201 ...

  4. 图论--最近公共祖先问题(LCA)模板

    最近公共祖先问题(LCA)是求一颗树上的某两点距离他们最近的公共祖先节点,由于树的特性,树上两点之间路径是唯一的,所以对于很多处理关于树的路径问题的时候为了得知树两点的间的路径,LCA是几乎最有效的解 ...

  5. P3379 【模板】最近公共祖先(LCA)

    P3379 [模板]最近公共祖先(LCA) 题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询 ...

  6. 洛谷P3379 【模板】最近公共祖先(LCA)(dfs序+倍增)

    P3379 [模板]最近公共祖先(LCA) 题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询 ...

  7. 「LuoguP3379」 【模板】最近公共祖先(LCA)

    题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下来N-1行每 ...

  8. 洛谷——P3379 【模板】最近公共祖先(LCA)

    P3379 [模板]最近公共祖先(LCA) 题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询 ...

  9. luogo p3379 【模板】最近公共祖先(LCA)

    [模板]最近公共祖先(LCA) 题意 给一个树,然后多次询问(a,b)的LCA 模板(主要参考一些大佬的模板) #include<bits/stdc++.h> //自己的2点:树的邻接链表 ...

随机推荐

  1. Python 3函数的参数冒号注释

    Python 3.7版本,函数的参数可以通过冒号来进行注释 def f(ham: str, eggs: str = 'eggs') -> str : print("Annotation ...

  2. 彰显个性│制作一个独一无二的动态 svg 头像

    一.头像预览 看一下博主的动态图像,是不是很炫酷,想不想拥有一个? 这是一个 svg 图片,svg 图片不仅可以通过制图软件制作外,其实也可以通过代码进行开发 因为 svg 本质上是一个下 xml 文 ...

  3. 老子云AMRT全新三维格式正式上线,其性能全面超越现有的三维数据格式

    9月16日,老子云AMRT全新三维格式正式上线,其性能远超现有的三维数据格式.目前已有含国家超算长沙中心.中科院空间所.中车集团等上百家政企事业单位的项目中使用了AMRT格式,大大提升了可视化项目的开 ...

  4. SAP HTLM Control

    HTML 事件 效果 代码 *&---------------------------------------------------------------------* *& Re ...

  5. hadoop集群搭建——单节点(伪分布式)

    1. 准备工作: 前提:需要电脑安装VM,且VM上安装一个Linux系统 注意:本人是在学习完尚学堂视频后,结合自己的理解,在这里做的总结.学习的视频是:大数据. 为了区分是在哪一台机器做的操作,eg ...

  6. freeswitch拨打分机号

    概述 电话语音服务中,有一种稍微复杂的场景,就是总机分机的落地场景,客户拨打总机号码之后,需要再拨打分机号转接到指定的话机. 分机号的拨打一般在总机接通之后,会有语音提示,总机收号之后转接分机. 分机 ...

  7. Day02 HTML语法

    有两种类型的标签 双标记<标记名>内容</标记名> 单标记<标记名/> 属性: 对标签的描述--属性,由属性名和属性值组成 <标记名 属性名 = " ...

  8. 深入理解 Java 对象的内存布局

    对于 Java 虚拟机,我们都知道其内存区域划分成:堆.方法区.虚拟机栈等区域.但一个对象在 Java 虚拟机中是怎样存储的,相信很少人会比较清楚地了解.Java 对象在 JVM 中的内存布局,是我们 ...

  9. MYSQL--安装2(命令行连接到数据库)

    使用命令行窗口连接 MYSQL 数据库[操作示意图]

  10. day02_基本语法

    基本语法 学习目标: 1. 数据类型 2. 变量 3. 编码介绍 4. 标识符和关键字 5. 字符串类型 6. 数据类型转化 7. 进制转换 8. 运算符 一.数据类型 什么是数据类型? 在开发软件的 ...