- 代码源挑战赛 Round 65
代码源挑战赛 Round 65 题解
- @ 2026-6-14 17:57:55
T1
100%
考场编号实际上就是 ,可以通过 得到。
座位编号为
时间复杂度 。
T2
100%
由于 ,我们可以枚举每个起点,然后检查从该起点开始的长度为 的环形子数组是否符合要求。判断是否为 的全排列,可以通过一个标记数组来标记出现过的 的数,并检查是否每个数都出现过即可。
时间复杂度 。(可以通过滑动窗口的方式实现 )
T3
100%
首先可以通过一次遍历可以计算出每个位置是否是峰顶(或谷底),可以设 表示 位置为峰顶,需要多次询问区间内的峰顶数量,即查询 数组在区间内 1 的个数,对 数组做前缀和即可 查询。
注意查询区间 时,实际需要统计的位置应为 ,必须保证 ,即 时需要特判。
时间复杂度 。
T4
100%
首先由于每个点的深度给定,因此树上每一深度所拥有的点是固定的,我们需要给每个点找一个父亲节点,使得最终的父亲数组 的字典序最小。
然后考虑一个点 (深度 ),对于所有深度为 的点,一定从中选择编号最小的且当前儿子个数未满的点作为 ;
其次对于同一深度的所有点,一定是编号越小的点先选择父亲,因为编号小的点在序列中更靠前,因此更需要选择编号更小的父亲。
综上,我们需要对所有点按深度排序,深度相同按下标排序,然后依次给每个点选择一个父亲,选择父亲时尽可能选择上一深度中编号最小的且还能选择的点。
然而暴力查找每个点的父亲最坏需要 的时间复杂度,可以利用双指针的思想,记录上一深度当前能选择的父亲的位置,每次被选择后更新该父亲的儿子个数,若儿子个数已满,则指针向后移动至下一个可选择的父亲,因此上一深度的每个点最多只会被枚举一次,时间复杂度降为 。
总的时间复杂度为 。
T5
100%
由于所有点的运动区间长度相同,均为 ,所有点的运动周期为 ,因此可以先令 ,然后可以将所有点分成两组:初始方向向右的和初始方向向左的。方向向右的所有点最终位置会 ,方向向左的所有点最终位置会 ,其中:
$$d= \begin{cases} t' & \text{如果 }t' \leq L \\ 2L-t' & \text{如果 }t' > L & & \end{cases}$$接下来我们就需要求两个数组,分别整体加上或减去同一个值 ,然后求这两个数组归并后的第 小值。
一个做法是可以先二分第 小的值 ,然后判断 的个数是否达到了 ,判断的过程可以在两个有序数组中分别二分即可求出 的个数,单次查询的时间复杂度 , 为值域。
另一个实现稍复杂但更快的做法是直接在其中一个有序数组中二分取的个数 ,那么另一个数组中取的个数即为 ,想要让这 个数作为前 小,那么就必须保证第一组取出的最大值 第二组未取的最小值,反之亦然。单次查询的时间复杂度为 。
T6
100%
直接枚举所有对 并计算 是困难的,但我们可以按贡献分解:
$$\operatorname{lcp}(u,v)=\sum_{L=1}^\infty\mathbf{1} [u和v的前L个字符相同] $$因此原式:
$$\sum_{u\in S}\sum_{v\in S}\mathrm{lcp}(u,v)=\sum_{u,v}\sum_{L=1}^\infty\mathbf{1}[u,v\text{ 前 }L\text{ 相同}] $$交换两个求和符号的顺序:
$$=\sum_{L=1}^\infty\left(\sum_{u,v}\mathbf{1}[u,v\text{ 前 }L\text{ 相同}]\right) $$固定一个长度 ,内层的 实际是要求有多少对 使得他们的前 个字符完全相同。
然后我们可以按实际内容分组:对所有可能的长度为 的序列 ,统计有多少子序列 以 为开头,记个数为 ,那么满足“前 个字符都是 ”的 的对数就是 。
把所有可能的 加起来:
$$\sum_{u,v}\mathbf{1}[u,v\text{ 前 }L\text{ 相同}]=\sum_{P\text{ 长度为 }L}\operatorname{cnt}(P)^2 $$最终原式即可转化为:
$$\sum_{u,v}\operatorname{lcp}(u,v)=\sum_{L=1}^\infty\sum_{P\text{ 长度为 }L}\operatorname{cnt}(P)^2 $$右边其实就是所有可能的前缀 的出现次数平方之和。
接下来我们需要求所有可能的前缀 的个数以及每个前缀的出现次数。
对所有非空子序列 按照结束位置分组(结束位置相同的 作为前缀出现的次数是相同的)。设 表示恰好以 位置为结尾的非空子序列个数,转移时考虑以 为结尾的子序列的上一个位置:
$$\mathrm{f}[i]=\sum_{j=\mathrm{prev}[i]}^{i-1}\mathrm{f}[j] $$其中 为上一个 出现的位置(第一次出现则为 ),再往前的位置无需转移,因为以 为结尾和以 为结尾的序列会产生重复的子序列。
转移可以通过前缀和优化为 。
现在我们求出了恰好以 位置为结尾的非空子序列个数 ,接下来要求以这 个子序列作为前缀的子序列个数,即从位置 开始的后缀数组中的所有不同子序列(包含空序列)。
类似地,可以从后往前dp,设 表示恰好以 位置为开始的非空子序列个数,并做后缀和后, 即为从位置 开始的后缀数组中的所有不同子序列。
最后答案:
$$\mathrm{Ans}=\sum_{i=1}^n\mathrm{f}[i]\cdot\left(suf[i+1]\right)^2 $$时间复杂度为 。