题面

有两棵

n

n

n 个点的有根树

T

1

T_1

T1​,

T

2

T_2

T2​,根是

1

1

1 ,共用编号

1

1

1~

n

n

n。求最大的点集

S

S

S 满足每个点在

T

1

T_1

T1​ 中一条到根的链上,且任意两个点在

T

2

T_2

T2​ 中没有祖先关系

输出该点集的大小,

t

t

t 组数据。

1

t

3

×

1

0

5

,

2

n

3

×

1

0

5

,

n

3

×

1

0

5

1\leq t\leq3\times10^5~,~2\leq n\leq 3\times 10^5~,~\sum n\leq 3\times 10^5

1≤t≤3×105 , 2≤n≤3×105 , ∑n≤3×105

题解

题面我已经转化了一部分了,继续转化:

我们处理出每个点在

T

2

T_2

T2​ 中的

d

f

s

\rm dfs

dfs 序,记为

d

f

n

[

i

]

{\rm dfn}[i]

dfn[i],并求出每个点在

T

2

T_2

T2​ 的子树内节点(包括自己)

d

f

n

[

j

]

{\rm dfn}[j]

dfn[j] 的范围,记为

[

l

i

,

r

i

]

[l_i,r_i]

[li​,ri​] ,不妨称它为点

i

i

i 的区间。

那么一个集合

S

S

S 可行,等价于集合中的点在

T

1

T_1

T1​ 一条到跟的链上,且集合中每个点的区间两两无交集

我们还可以发现一些性质:每两个点的区间要么无交集,要么存在包含关系,且一个左端点只对应一个右端点。那么我们就可以想出一个贪心策略:互相包含的区间,取较小的保留。把它挪回树上,其实就相当于存在一个点的子孙可选的时候该点不选最优一样。

接下来就简单了。我们只需要从

T

1

T_1

T1​ 的根开始往下

d

f

s

\rm dfs

dfs ,每到一个点往某数据结构中加入自己的区间:

  • 如果被数据结构中原有的更大区间包含了,那么把那个大区间删掉,把自己加进去。
  • 如果包含了数据结构中原有的至少一个小区间,那么不加自己。
  • 否则,把自己加入进去,此时数据结构大小(即答案)+1。

然后遍历儿子。

回溯的时候数据结构要退回来时的状态。

具体用的数据结构就五花八门了,比较懒的可以直接用 set 之类,稍微比较聪明的可以打打树状数组或常数优秀的 zkw 线段树(有的人利用了两棵树中父亲编号小于儿子编号的特性,打出最优秀的三行朴素树状数组跑了

R

a

n

k

1

\rm Rank~1

Rank 1)。

CODE

下面是个比较好看懂的 set 代码

#include<set>
#include<cmath>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 300005
#define ENDL putchar('\n')
#define LL long long
#define DB double
#define lowbit(x) ((-x) & (x))
#define SI set<int>::iterator
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;
vector<int> g0[MAXN];
int L[MAXN],R[MAXN],lR[MAXN],tim;
void dfs0(int x,int ff) {
L[x] = ++ tim;
for(int i = 0;i < (int)g0[x].size();i ++) {
if(g0[x][i] != ff) dfs0(g0[x][i],x);
}R[x] = tim; lR[L[x]] = R[x];
return ;
}
vector<int> g[MAXN];
int d[MAXN],dfn[MAXN],rr[MAXN],cnt,ans;
set<int> st;
void dfs(int x,int ff) {
int ad = 0;
if(st.empty()) st.insert(L[x]);
else {
SI i = st.lower_bound(L[x]);
if(i != st.begin()) {
i --;
if(lR[*i] >= R[x]) ad = *i,st.erase(ad),st.insert(L[x]);
else {
i ++;
if(i == st.end() || *i > R[x]) st.insert(L[x]);
}
}
else if(i == st.end() || *i > R[x]) st.insert(L[x]);
}
ans = max(ans,(int)st.size());
for(int i = 0;i < (int)g[x].size();i ++) {
if(g[x][i] != ff)
dfs(g[x][i],x);
}
if(st.find(L[x]) != st.end()) st.erase(L[x]);
if(ad) st.insert(ad);
return ;
}
int main() {
int T = read();
while(T --) {
n = read();
tim = 0; cnt = 0;
st.clear();
for(int i = 1;i <= n;i ++) {
g0[i].clear();g[i].clear();lR[i] = 0;
}
for(int i = 2;i <= n;i ++) {
s = read(); g[s].push_back(i);
}
for(int i = 2;i <= n;i ++) {
s = read(); g0[s].push_back(i);
}
dfs0(1,0);ans = 0;dfs(1,0);
printf("%d\n",ans);
}
return 0;
}

CF1528C Trees of Tranquillity(图论,数据结构)的更多相关文章

  1. [ An Ac a Day ^_^ ] hdu 1662 Trees on the level 数据结构 二叉树

    紫书上的原题 正好学数据结构拿出来做一下 不知道为什么bfs的队列一定要数组模拟…… 还可以练习一下sscanf…… #include<stdio.h> #include<iostr ...

  2. 图论&数据结构——并查集

    Wikioi 4246 NOIP模拟赛Day2T1 奶牛的身高  题目描述 Description 奶牛们在FJ的养育下茁壮成长.这天,FJ给了奶牛Bessie一个任务,去看看每个奶牛场中若干只奶牛的 ...

  3. 2021record

    2021-10-14 P2577 [ZJOI2004]午餐 2021-10-13 CF815C Karen and Supermarket(小小紫题,可笑可笑) P6748 『MdOI R3』Fall ...

  4. Noip 训练指南

    目录 Noip 训练指南 图论 数据结构 位运算 期望 题解 Noip 训练指南 目前完成 \(4 / 72\) 图论 [ ] 跳楼机 [ ] 墨墨的等式 [ ] 最优贸易 [ ] 泥泞的道路 [ ] ...

  5. TokuDB性能测试报告

    一 .背景介绍 近年来,TokuDB作为MySQL的大数据(Big Data)存储引擎受到人们的普遍关注.其架构的核心基于一种新的叫做分形树(Fractal Trees)的索引数据结构,该结构是缓存无 ...

  6. Disruptor——一种可替代有界队列完成并发线程间数据交换的高性能解决方案

    本文翻译自LMAX关于Disruptor的论文,同时加上一些自己的理解和标注.Disruptor是一个高效的线程间交换数据的基础组件,它使用栅栏(barrier)+序号(Sequencing)机制协调 ...

  7. NOIP考点

    NOIP考点 基础算法 图 树 数论 数据结构 动态规划 搜索 其他算法 省选知识点汇总 图论 数据结构 字符串相关算法及数据结构 数学 计算几何 搜索 动态规划 其他算法 转自:巨佬的博客 加*号是 ...

  8. Luogu P3783 [SDOI2017]天才黑客

    题目大意 一道码量直逼猪国杀的图论+数据结构题.我猪国杀也就一百来行 首先我们要看懂鬼畜的题意,发现其实就是在一个带权有向图上,每条边有一个字符串信息.让你找一个点出发到其它点的最短路径.听起来很简单 ...

  9. Marked Ancestor [AOJ2170] [并查集]

    题意: 有一个树,有些节点染色,每次有两种操作,第一,统计该节点到离它最近的染色父亲结点的的号码(Q),第二,为某一个节点染色(M),求第一种操作和. 输入: 输入由多个数据集组成.每个数据集都有以下 ...

随机推荐

  1. 技术分享 | Appium环境安装与架构介绍

    原文链接 Appium架构 Appium 设计哲学 不需要为了自动化而重新编译或修改被测应用 不应该让移动端自动化测试限定在某种语言或者某个具体的框架 不要为了移动端的自动化测试而重新造轮子 移动端自 ...

  2. 如何写出同事看不懂的Java代码?

    原创:微信公众号 码农参上,欢迎分享,转载请保留出处. 哈喽大家好啊,我是没更新就是在家忙着带娃的Hydra. 前几天,正巧赶上组里代码review,一下午下来,感觉整个人都血压拉满了.五花八门的代码 ...

  3. 不花钱~Python制作视频解析免费追剧神器

    同学们在闲暇之余是否喜欢看电影或者电视剧呢? 今天带领大家使用python制作能免费追剧的桌面软件.还在等什么?发车了! 效果我就不再这里演示了https://jq.qq.com/?_wv=1027& ...

  4. NC16597 [NOIP2011]聪明的质监员

    NC16597 [NOIP2011]聪明的质监员 题目 题目描述 小T 是一名质量监督员,最近负责检验一批矿产的质量.这批矿产共有 \(n\) 个矿石,从 \(1\) 到 \(n\) 逐一编号,每个矿 ...

  5. 基于 Github Actions 自动部署 Hexo 博客

    前言 前不久使用了 Hexo 搭建独立博客,我是部署在我的腾讯云轻量应用服务器上的,每次都需要 hexo deploy 然后打包.上传.解压和刷新 CDN,非常麻烦.我的服务器配置也不高 2C2G 无 ...

  6. manjaro 安装后的基本配置

    第一步:设置官方镜像源 sudo pacman-mirrors -i -c China -m rank # 输入以上命令后会有弹出框,选择一个国内镜像(推荐 https://mirrors.ustc. ...

  7. 攻防世界MISC—进阶区11-20

    11.János-the-Ripper 得到未知类型的文件,010 Editor打开后看到pk,得知是真加密的zip文件. 密码在文件中没有提示,根据题目名字,János-the-Ripper Ján ...

  8. 「Python实用秘技09」更好用的函数运算缓存

    本文完整示例代码及文件已上传至我的Github仓库https://github.com/CNFeffery/PythonPracticalSkills 这是我的系列文章「Python实用秘技」的第9期 ...

  9. Netty源码解读(一)-前置准备

    前置条件 源码版本netty4.1 了解Java NIO.Reactor模型和Netty的基本使用. 解释一下: Java NIO:了解BIO和NIO的区别以及Java NIO基础API的使用 Rea ...

  10. 使用codeblocks创建新项目

    很多同学在学习C或C++版的数据结构的时候,自己写项目是一个不错的锻炼方法,而用codeblocks写项目的时候我们就会遇到很多问题了,比如说: 1.如何建立新项目. 2.如何建立头文件和主函数文件. ...