UOJ Logo vfleaking的博客

博客

UOJ Round #3 题解

2014-12-21 22:36:52 By vfleaking

核聚变反应强度

算法一

对于 n=1 的数据,就是求一个数次大的约数。

众所周知一个数x的约数是成对出现的(dxd),其中总有一个不超过x。所以从1a1地枚举d就能找出所有a1的约数了。排序输出次大的即可。

复杂度:O(a)

算法二

先找出a1的所有约数,然后枚举isgcd(a1,ai)显然也是a1的约数,所以枚举a1的所有约数,找到是ai约数的次大的即可。

复杂度:O(na)

算法三

考虑分解质因子后:

a=p1x1p2x2...pmxm

b=p1y1p2y2...pmym

则:gcd(a,b)=p1min(x1,y1)p2min(x2,y2)...pmmin(xm,ym)

我们发现,ab的公约数都一定是gcd(a,b)的约数。那么为了得到次大公约数,只需求出gcd(a,b),再除去一个最小的公共质因子即可。

a1O(a1)的时间分解得到O(log(a1))个质因数,每次对于ai,先求出g=gcd(a1,ai),然后枚举a1的每个质因数,找到最小的能整除g那个,设其为pg/p即为所求。(不存在则为输出1

复杂度:O(a+nlog(a))

一个骗分算法

考虑算法二,我们预先对 a1 的约数们排好序,然后枚举 i,从约数表里每次二分到 gcd(a1,ai)所在位置,再往前枚举,找到第一个能整除ai的即为次大公约数。

虽然复杂度不靠谱,但是对于ai1012的范围实际运行速度十分优秀。需要构造针对的数据才能卡住。

还有另一个骗分算法,求出 gcd 然后暴力枚举最小质因子。好多人写这个啊……你们都没意识到复杂度不对么……放你们一马给了 80 分。

(有这种闲心的为啥不写正解啊,你们考虑过 maker 的感受吗!QAQ)

铀仓库

算法一

我们先来证明几个显然的结论。

结论一:小O的行动一定是,每次看准一个箱子,从s跑过去拿起来,然后直接搬回s

首先,小O一定不会把一个箱子搬到离s更远的地方。其次我们考虑,如果小O进行了这样的动作:从a搬到b、从c搬到d,其中a<b<c<d<s,那么等价于从a搬到d,从c搬到b,显然干了多余的事。

结论二:起始位置s一定有箱子。

考虑确定了s,一定是搬来前若干近的箱子,也就一定是连续的一段,所耗时间为距离和的两倍。我们知道,给定数轴上若干个点,选一个坐标最小化每个点到它的距离和,那么一定是选这些点坐标中的中位数(调整法可证)。

说了一大堆废话,我们得到一个暴力做法。枚举s,每次拿来一个最近的即可。

复杂度:O(n2T)

算法二

枚举s,由于是取前若干近的过来,我们二分取的最远的在哪。如果xi=i那么直接可以O(1)确定左右端点,否则我们需要再一次二分确定左右端点。

复杂度:O(nlognlogX)X表示坐标范围。

算法三

先二分答案,于是问题变成了:求叠到K个箱子所需的最短时间。

由于取的一定是连续的一段,设其为[l,r],若ai=1,我们直接从左到右枚举s,此时由于左边的越来越远,右边的越来越近,lr一定是非降的。所以,每次s右移的时候,判断r+1处是否比l近,如果是就一直替换,直到不是为止。

移动左右端点的同时顺便维护一下距离和就好了。

这样的复杂度是O(n+Σai)的,还是不能通过全部数据。

加一个小优化就好了。把每个位置的ai个箱子想象成横着紧贴成一排,称为一组箱子,那么每次我们移动左右端点lr的时候,事实上是干了很多重复的事情的。

如果lr所在箱子组都不变,那么会一直进行替换,直到某个端点换组。所以在每次替换的时候,直接加速到某个端点换组的时刻即可。

复杂度:O(nlogX)

链式反应

算法一

按照题意进行爆搜,可以通过第一个测试点获得 10 分。

算法二

我们得思考一下这题题意到底是啥意思。其实说,一棵有标号的树,编号满足堆的性质,对于儿子方向每个结点有两条红色边,c 条黄色边,c 属于集合 A,统计方案数。

所以我们用 f[n] 表示 n 个结点的这个样子的树的方案数。然后要么 1 号点没有进行核裂变,此时必有 n=1f[n]=1。要么 1 号结点进行了核裂变,此时我们枚举两个中子打中的结点的子树的大小 j,k,剩下的就是被破坏死光打中的,于是要满足 i1jkA。然后考虑两个中子打中的结点的子树,我们先选出它们的编号,方案数是 (i1j)(i1jk),然后我们就可以安全地把这两棵子树的结点都重标号,方案数自然是 f[j]f[k]。所以我们把这些杂八事综合起来就得到了一个简单的DP:f[i]=jk(i1j)(i1jk)f[j]f[k]。这里的 j,k 要满足 i1jkA。直接暴力递推就得到了一个 O(n3) 的算法。

什么,过不了样例?当然过不了样例了,因为有重复计数。那两个中子是等价的,一个方案自然会被统计了两遍,只要把 DP 方程除以 2 就好了。

可以通过前 4 个测试点获得 40 分。

算法三

其实只要在算法二基础上优化就行了。我们仔细观察式子:(i1j)(i1jk)=(i1)!j!k!(ijk1)!。有四个部分,第一个只跟 i 有关,第二个只跟 j 有关,第三个只跟 k 有关,第四个只跟 ijk1 有关。所以我们可以递推一个 g[i]=kf[k]k!f[ik](ik)!。然后递推 f[i] 时我们枚举 j+k 的和 s,那么直接读取 g[s] 的值,然后剩下的部分就是一些跟 is 有关的了。

好像说得挺意识流的,具体就是:f[i]=12sg[s](i1)!(is1)!s 要满足 is1A

可以通过前 6 个测试点获得 60 分。

算法四

再观察一下式子,搞个数组 a,其中如果 iAai=1i!,否则为 0。然后原递推式的右边的第 i 项其实就是 ga 的卷积的第 i1 项然后再乘以 (i1)!。然后其实 g 本身也是个卷积,就是 f[i]i! 这个数列的平方。

我们可以使用分治。每次分治一个区间 [l,r],我们找出中点 m,然后递归 [l,m],然后求出 [l,m][m+1,r] 的贡献,再递归右边。

考虑左边对右边的贡献,“(i1j)(i1jk)f[j]f[k]” 中,左边可能作为 f[j] 也可能作为 f[k] 出现,也可能都出现。我们只要考虑这几种情况然后分别进行 FFT 就行了。看起来要和整个 a 或者 f 进行卷积?其实不然,只用取出一个长度为分治时的区间的长度的前缀就可以了。

时间复杂度 O(nlog2n)。可以通过前 9 个测试点获得 90 分。常数小的话应该能过。

算法五

以下内容需要一点微积分知识……如果是微积分恐惧症……您还是看算法四加卡常数吧~我还是很良心的没让所有人都非得用算法五才能AC~

嗯,既然都发现是卷积了,那么肯定少不了生成函数。我们记 f[i]i! 的生成函数为 x(t)ai 的生成函数为 a(t),那么生成函数就要满足:x(t)=12a(t)x2(t)+1

拨一下 mathematica 就会发现它解不出来这个微分方程,所以只有自力更生了。

首先科普下一般来说应该怎么解一个多项式方程(更严谨地说这里应该是形式幂级数)。假设 x(t) 满足 f(x(t))=0,那么我们先求出 0 次项的系数,然后每次倍增。也就是说每次我们有一个多项式 xn 满足 xnx 的前 n 项系数是一样的,我们记为 xxn(modtn),然后我们要求出 x2n 满足 xx2n(modt2n)。使用泰勒展开可以得到:

0=f(x2n)=f(xn)+f(xn)(x2nxn)+12f(xn)(x2nxn)2+

注意到如果只考虑前 2n 项,那么就可以去掉二次项及之后的项然后解出:

x2nxnf(xn)f(xn)(modt2n)

由于 T(n)=T(n/2)+O(nlogn) 解出来是 T(n)=O(nlogn),所以只要能足够快地把 x 带入 f(x),那么就能 O(nlogn) 解方程。(当然需要 FFT 做多项式乘法)

什么这里有个除法?我们可以利用牛顿迭代求出一个形式幂级数的乘法逆元,于是就能做除法了。

于是微分方程也可以如法炮制,假设有个这样的一阶微分方程:

ddtx=f(x)

我们还是老样子:

ddtx2nf(xn)+f(xn)(x2nxn)(modt2n)f(xn)+f(xn)x2nf(xn)xn(modt2n)

所以问题变成了这玩意儿怎么解……而这玩意儿其实很好解……我们两边同时乘以 ef(xn)dt(囧……由于公式嵌套过多,排版已经开始混乱了),记作 r

ddtx2nr(f(xn)f(xn)xn)r+f(xn)x2nr(modt2n)ddt(x2nr)(f(xn)f(xn)xn)r(modt2n)

然后只要把右边积分再除以 r 就行了。

注意到虽然人类无法方便地给任意函数积分,但是给多项式求导和求积分简直易如反掌。所以唯一的难点在于 r 怎么求。

好吧其实我几个月前 YY 到这里直接就弃疗了,后来翻了翻论文,知道了怎么求 ex。由于 ex 的反函数是 ln(x)ln(x)t 的导数是 xx,所以 ln(x) 可以快速求,然后我们就可以进行牛顿迭代求 ln(x) 的反函数得到 ex

对于本题,f(x)=12ax2+1,然后无脑使用上述方法就行了。

本来还想当一个原创算法呢……结果后来发现国外论文上早就有了……果然我还是太naive……

时间复杂度 O(nlogn)。可以通过所有测试点获得 100 分。

形式幂级数真是个优美的东西啊,很多在实数域内有条件收敛的算法,到形式幂级数上就一定收敛了,我不得不表示赞叹。

评论

沙发
板凳
前排
A题好水。。。一眼题
评论回复
Picks:跪…
tm:回复 @Picks:跪。。。
最后一次用Pascal,竟然惨成这样。。。被语言坑啊。。。TAT
怒赞~\(≧▽≦)/~
Rating掉得真开心。
评论回复
thomount:呵呵,不要和我比。。。
感觉我的OI生涯就是拿来搞笑的。。
远距离orz!
orz
好评
UOJ会成为FFTOJ么...3发比赛两题FFT
orz
真棒!!
orz
@vfleaking 您的标程都跑了3s+,算法四多个log真的能卡进4s内么?
@hzyfr 这个真的可以……因为常数小……裸写的话大概是4秒多一点,然后分治时注意到有很多次dft是对着同一个数组的……优化下…… 然后还可以利用fft是循环卷积的性质……所以你可以把“dft两次再idft”的模式扔进垃圾桶……预先开个比较大的数组,然后狂dft呀乘起来什么的,到最后再idft,中间任由它爆(因为反正最后我们只需要知道乘积的后半段……)
评论回复
hzyfr:还真是巧妙啊,卡常数果然是一门艺术
有一个地方看不懂,就是求多项式,为什么会涉及到ex,然后这东西又怎么变成多项式?
@hzyfr 其实他们是形式幂级数。你可以粗略地感受一下,f(z)g(z) 加减乘除还是形式幂级数(除法时0是个例外 = =……)。。他们还能求导求积分。。 这意味着可以像实数差不多的样子用……(说白了就是形式幂级数带上加和乘形成了一个域,且能在上面定义导数什么的……) 所以微积分的相关玩法还是成立的。 随便给个不奇葩的函数展开成形式幂级数的方法差不多就是泰勒展开就行了。比如 ez=k0zkk!…… 然后随便给两个 fg,我们可以把他们套起来 f(g(z))。 所以 ef(z) 的意思就很明显了。
@vfleaking 上述题解中间,提及“两边同时乘以 e−∫f′(xn)dte−∫f′(xn)dt”,请问为什么要乘这个? 下面式子中怎样消去的最后一项?
评论回复
vfleaking:这是微分方程的求解技巧,你可以找一本微积分学习一下 233

发表评论

可以用@mike来提到mike这个用户,mike会被高亮显示。如果你真的想打“@”这个字符,请用“@@”。