ecfinal2019热身赛B题
Posted cdcq
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了ecfinal2019热身赛B题相关的知识,希望对你有一定的参考价值。
原题:
给你一个长度为1e5的序列ai,问你它的所有子序列的最大值与最小值之差的1000次方的和是多少
即∑_{p是a的子序列}(max{p}-min{p})^1000
这题难点在于(max-min)^1000这个东西外边套了次方,不太好处理
首先需要注意一点的性质是因为只考虑最大值和最小值,所以考虑子序列实际上是考虑集合的子集
那么可以对原序列排序,然后枚举两个端点,对应区间的包含端点的所有子集就是最大值和最小值分别为端点的子集
即此区间对答案的贡献为((a[j]-a[i])*2^(j-i-1))*1000
接下来处理(max-min)^1000
一个想法是差分转差为和,然而并没有什么卵子用
另一个想法是枚举区间时不枚举端点,而是枚举长度,这样可以把2的次幂提出来
但是仍旧无法把(max-min)^1000展开
最后的做法是暴力把上式展开,或者手玩小数据,可以发现规律
(涉及到区间问题时常用的一个思路是固定一个端点,考虑与另一个端点有关的变化情况)
所有左端点为1的区间的贡献和为
∑((a[j]-a[1])*2^(j-2))*1000
=2^1000*(a[2]-a[1])^1000+2^1001*(a[3]-a[1])^1000+...+2^(1000+n-1)*(a[n]-a[1])^1000
=2^1000*((a[2]-a[1])^1000+2*(a[3]-a[1])^1000+...+2^(n-1)*(a[n]-a[1])^1000)
把2^1000提出来,令S=(a[2]-a[1])^1000+2*(a[3]-a[1])^1000+...+2^(n-1)*(a[n]-a[1])^1000
暴力展开1000次方
=(a[2]^1000+1000*a[2]^999*a[1]+...+a[1]^1000)+2*(a[3]^1000+...+a[1]^1000)+...+2^(n-1)*(a[n]^1000+...+a[1]^1000)
合并同次项
=(a[2]^1000+2*a[3]^1000+4*a[4]^1000+...+2^(n-1)*a[n]^1000)
-1000*a[1]*(a[2]^999+2*a[3]^999+4*a[4]^999+...+2^(n-1)*a[n]^999)
+C(1000,2)*a[1]^2*(...)
...
+a[1]^1000*(1+2+4+...+2^(n-1))
发现规律了木有!
随着右端点下标增加,同次项的系数每次乘2
而a[j]^0,a[j]^1,a[j]^2,...,a[j]^1000可以预处理
那么如果从左往右枚举左端点i,(a[i+1]^k+2*a[i+2]^k+...+2^p*a[n]^k)可以快速由(a[i+2]^k+...+2^(p-1)*a[n]^k))乘2+a[i+1]^k得到
C可以预处理,a[i]^k也预处理好了,那么对于每一个左端点我们都可以1000次O(1)操作得到它对答案的贡献
耗时1e5*1e3,有点紧,要善用递推性质,盲目龟速乘会挂
还需要注意的一个问题是序列中的相同元素
实际上手玩小数据或者直接证明都可以发现,相同的数按照上述方法处理仍然能得到正确的结果
这个留给读者证明233
没有代码
以上是关于ecfinal2019热身赛B题的主要内容,如果未能解决你的问题,请参考以下文章
2018-2019 ACM-ICPC ECfinal I. Misunderstood … Missing
2020中国高校计算机大赛·华为云大数据挑战赛--热身赛--赛题分析数据理解