CF1294F 题解

博客探讨了如何使用O(nlogn)的时间复杂度解决CF1294F题目,涉及树的直径问题。作者首先解释了为何两个特定点必须是直径的端点,然后转化为寻找一个点,使得该点到直径两端点的路径与直径并行时,经过的边数最多。通过预处理深度信息和计算最近公共祖先(LCA),博主给出了求解每个点的答案的方法,并最终找出最大值作为最终答案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

传送门(洛谷)

在这里插入图片描述
看到大佬们都是用的 O ( n ) O(n) O(n) 的做法,本蒟蒻不会,就按合情合理的思路写了一个 O ( n log ⁡ n ) O(n\log n) O(nlogn) 的做法。

首先可以确定,其中的两个点肯定是直径的两个端点。感性理解一下,如果不是的话,那么把其中的点换成直径的一个端点,答案一定不会更差,因为直径已经是树上的最长路径了。然后问题就变成了在树上找到一个点,使它到直径的两个端点的路径和直径的并所经过的边数最多。我们考虑一个简单的不能算容斥的容斥。

在这里插入图片描述
拿这棵树来说,假如直径的两个端点是 1 1 1 6 6 6,我们要找 5 5 5 这个点,那么分别记录下来直径的两个端点到其余点的距离,然后我们将其中的一个端点设为根,把其余点的深度和求 l c a lca lca 用的数组预处理出来。那么我们可以求出 5 5 5 和深度较深的那个端点的 l c a lca lca,然后就是一个小的容斥,假设两个端点到 5 5 5 的距离分别为 d i s 1 dis1 dis1 d i s 2 dis2 dis2,那么答案就是 d i s 1 + d i s 2 + d e p [ 5 ] − d e p [ l c a ] dis1+dis2+dep[5]-dep[lca] dis1+dis2+d

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值