清华月赛 Yazid的新生舞会题解

本文介绍了处理特定数值序列的六种算法,针对不同规模和类型的输入数据优化了时间复杂度,从O(n^3)到O(nlogn)。算法覆盖枚举区间求众数、使用线段树优化计算等技巧。

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

算法一(针对n300的测试点)

考虑枚举所有区间,然后求其众数及出现次数,并判断是否超过区间总长的一半,统计答案即可。时间复杂度O(n3)

算法二(针对n2,000的测试点)

考虑先枚举区间的左端点l,再从左到右枚举右端点r并用数组维护每个数的出现次数,同时使用变量维护当前众数、众数的出现次数。不难发现,当右端点向右移动时,这些信息都是非常方便维护的。于是我们便可以在O(n2)的时间复杂度内统计所有答案。

算法三(针对type=1的测试点)

对于01序列,我们不难发现,众数出现次数严格大于子区间长度当且仅当子区间内0,1出现次数不同(那么那个出现次数较多的就是众数)。于是我们将原序列中的0视作1,并对该序列求前缀和S,则子区间[l,r]为“新生舞会的”,当且仅当Sl1=Sr,因此对S进行排序并从小到大枚举便可求出答案。

算法四(针对type=2的测试点)

对于所有数的出现次数都较小(不超过15)的情况,不难发现所有“新生舞会的区间”的长度也会较小(不超过2×151=29)。于是使用算法二,枚举所有长度少于30的区间并统计答案即可。

算法五(针对type=3的测试点)

对于Ai7的测试点,我们不妨枚举所有值作为众数的情况。考虑统计所有众数为k的“新生舞会的”区间,将所有等于k的位置取为1,不等于k的位置取为1,得到新序列B并求前缀和得到序列S,则区间[l,r]需要被统计,当且仅当SrSl11。从左到右枚举右端点,并用线段树维护当前右端点左边每种前缀和出现的次数即可。时间复杂度O(8nlogn)

算法六(针对所有测试点)

考虑改进算法五。我们考虑取出所有B中的极长1子区间,观察这些区间中的所有点作为右端点对答案的贡献。不难发现极长1子区间[l,r]中的所有点作为右端点对答案的贡献为rl+1i=1Sl1i1j=cnt[j],其中cnt[j]表示在区间[0,l1]之间前缀和为j的端点数目;在统计这段区间的答案后,我们还需要对区间[Si1(rl+1),Si11]中的所有cnt均进行+1操作。显然地,我们使用一个维护BiCi=i×Bi的线段树就可以支持这些查询、修改操作。于是我们使用这棵线段树维护相关信息,并从左到右枚举右端点,统计答案即可。

不难发现,极长1子区间的数目与序列中k的数目同阶,因此,对于统计众数为k的“新生舞会的”区间的时间开销,不难发现我们通过上述优化将时间开销缩短到了O(Aklogn)

于是,总时间复杂度即为O(kAklogn),即O(nlogn)

n100,000测试点的说明

对于常数较大的与标准算法时间复杂度相同的算法,以及一些时间复杂度略大于标程的算法,可能存在无法通过所有测试点的情况。这类算法可以通过这类测试点。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值