【题解】2020牛客NOIP赛前集训营-普及组(第一场)
before contest
这套题并没有涉及太多的算法或者数据结构。
更多的还是偏思维和基础。
本场隔壁提高组也是本人负责,感兴趣可以过去看一看。
T1 牛牛的密码
签到题,暴力模拟,考察语言基础。
100pt
输入字符串循环遍历处理,由于我们知道字符的ASCII码中,小写英文字母,大写英文字母,数字,都是连续的。
所以可以设计一个分支结构判断,这样做的好处是可以最后用else来表示其他特殊字符。
T2 牛牛的跳跳棋
思维题,简单贪心
20pt
dfs爆搜之
40pt
如果碰到连续的1,肯定对最后一个出现的1把它变成2,然后如果后面还是0,就改成1。
比如"0 0 1 1 0 1 1 1 0 0 0"修改后变成"1 1 1 2 0 1 1 2 0 1 1"。
(换句话说就是修改后变成连续若干个1,把末尾的1变成2,后面接一个0,然后依次类推,最后变成1 1 1...1 1 2 0...的模式)
100pt
原本不要求修改,质问你能不能到达的话是一个青蛙跳的经典问题。
原题是这样做的,首先贪心的想,往回跳是不必要的,因为如果可以走到第i个格子,那么必定可以在最后一次跳格子的时候力度小一点走到第i-1个格子。
这样的话易知在不修改数字的情况下,从第一个格子开始走可以到达的格子是连续的。
所以这样设计算法,记录一个r,表示当前能够到达的最靠右的格子。
然后遍历过去,更新r=max(p[i]+i,r)。过程中发现i>r的情况就说明不可达,反之说明可达。
本题其实还是相同的思路,只不过在更新右边界的同时,我们还需要知道右边界是谁提供的,一边更新r,同时储存更新右边界的格子pos。
一旦发现i>r的情况,首先让a[pos]+1。因为首先pos提供了当前右边界,所以当a[pos]+1时可以立即更新边界r,而其他格子即使增加也不一定能够使得边界扩充,这就保证了题目中的“修改次数最少”这个要求,同时由于是顺序遍历,pos是第一个达到右边界r的格子,所以也同时保证了“最小字典序”这个要求。
在更新后我们令pos=i,然后继续遍历,这样在算法结束后就得到了一个“最小操作次数并且最小字典序”的操作序列。
T3 牛牛的最大兴趣组
简单数学题
20pt
dfs爆搜之
40pt
40pt是一个二分图染色算法。虽然普及组不考图论,但是二分图染色就是个dfs。
首先两重for循环枚举i,j,如果i*j是某个数字的三次方,就把i和j连一条边。
然后对于每个联通块进行二分图染色,假设红色有a个节点,蓝色有b个节点。那么就令ans+=max(a,b)。
100pt
对于任何一个正整数都可以表示成的形式,其中是最大的立方因子。
如果a是n去掉最大立方因子后的结果,b是m去掉最大立方因子后的结果,如果nm是立方数,则ab也是立方数,且a和b是唯一配对的。
例如2,16,54,686在去除立方因子后全部等于2,然后4,32,108,1372在去除立方因子后全部等于4。
所以{2,16,54,686}中的任意一个数字乘上{4,32,108,1372}中的任意一个数字都是一个立方数。
一开始输入数字,筛除掉它们的立方因子,然后统计每种数字出现次数。
然后对于每一对出现的种类,都是两两互斥的,所以贪心的取出现次数较多的部分即可。
在实际处理上是有细节的,筛除立方因子可以只枚举立方根以内的质数,这样的话单次去除立方因子的复杂度可以压到200次运算左右。
T4 牛牛的滑动窗口
简单数据结构,简单算法题
大概普及T3,T4会稍微上一点难度。
一般有个说法,题目标题中出现的算法名都是用来骗人的。
这个题的话,有一半是骗人的吧,写个划窗能拿一半分。
但是std也确实是从滑动窗口算法得到的启示。
30pt
暴力三重for循环。
50pt
真的写划窗或者RMQ啥的优化掉一层for
100pt
滑动窗口的使用条件正如题目中介绍的那样,需要保证每个查询的端点值是“双单调”的。
实际上划窗算法,或者叫尺取法。的使用条件就是“双单调”。(实际上有部分选手懒得管它到底几个单调,一般看见一个单调就直接二分上了,反正没人卡log)
首先需要掌握一个技巧,“单调队列O(n)求滑动窗口极值”,这个算法。
如图,假设这是一个用划窗扫描数组5,7,4...的场景,假设求最大值。
在继续扫描到7的时候,我们想一下5这个数据有没有用。
因为是滑动窗口求最大值,当前的最大值已经是7了,它有没有可能再变回5?
显然不可能,因为滑动窗口要求端点递增,所以5将会先比7被舍弃。
接下来继续扫描,4要不要保留,显然保留,因为根据滑动窗口的定义,7一定比4先被舍弃。
然后当7被划窗划过后,4上位变成当前最大值。
有了这个算法以后,我们开两个单调栈,分别维护最大值和最小值,然后从左到右遍历过去。
这样的话就可以知道当r单调向右滑动的时候,l端点落在不同的位置时最大值最小值各是多少。
由于值域不超过100,那么栈内的元素数目肯定是不会多于100的。
所以我们可以暴力穷举栈内元素。
然后因为两个单调栈都是有序的,在这里再次借助双指针划窗。提取出max*min相同的区间长度是L到R。
使用差分维护一个数组a,a[k]表示区间长度为k的答案。
然后将使用差分将a[L]~a[R]这段区间全部加max*min即可。
最后对差分数组求前缀和,就得到了每一个区间长度的答案。
after contest
没提前准备大样例,最近比较忙忘了这回事了。