简介
常系数齐次线性递推数列(又称为 C-finite 或 C-recursive 数列)是常见的一类基础的递推数列。
对于数列 ( a j ) j ≥ 0 和其递推式
a n = j = 1 ∑ d c j a n − j , ( n ≥ d )
其中 c j 不全为零,我们的目标是在给出初值 a 0 , … , a d − 1 和递推式中的 c 1 , … , c d 后求出 a k 。如果 k ≫ d ,我们想要更快速的算法。
这里 ( a j ) j ≥ 0 被称为 d 阶的常系数齐次线性递推数列。
Fiduccia 算法
Fiduccia 算法使用多项式取模和快速幂来计算 a k ,时间为 O ( M ( d ) log k ) ,其中 O ( M ( d )) 表示两个次数为 O ( d ) 的多项式相乘的时间。
算法 :构造多项式 Γ ( x ) := x d − ∑ j = 0 d − 1 c d − j x j 和 A ( x ) := ∑ j = 0 d − 1 a j x j ,那么
a k = ⟨ x k mod Γ ( x ) , A ( x ) ⟩
其中定义 ⟨ ( ∑ j = 0 n − 1 f j x j ) , ( ∑ j = 0 n − 1 g j x j ) ⟩ := ∑ j = 0 n − 1 f j g j 为内积。
证明 :我们定义 Γ ( x ) 的友矩阵为
C Γ := 1 ⋱ 1 c d c d − 1 ⋮ c 1
我们定义多项式 b ( x ) := ∑ j = 0 d − 1 b j x j 和
B b := [ b 0 b 1 ⋯ b d − 1 ] ⊺
观察到
C Γ 1 ⋱ 1 c d c d − 1 ⋮ c 1 B b b 0 b 1 ⋮ b d − 1 = B x b mod Γ c d b d − 1 b 0 + c d − 1 b d − 1 ⋮ b d − 2 + c 1 b d − 1
且
C Γ ( C Γ ) 2 ⋯ ( C Γ ) k = [ B x mod Γ B x 2 mod Γ ⋯ B x d mod Γ ] , = [ B x 2 mod Γ B x 3 mod Γ ⋯ B x d + 1 mod Γ ] , = [ B x k mod Γ B x k + 1 mod Γ ⋯ B x k + d mod Γ ]
我们将这个递推用矩阵表示有
a k a k + 1 ⋮ a k + d − 1 = ( ( C Γ ) ⊺ ) k = ( ( C Γ ) k ) ⊺ c d 1 c d − 1 ⋱ ⋯ 1 c 1 k a 0 a 1 ⋮ a d − 1
可知 ( ( C Γ ) k ) ⊺ 的第一行为 B x k mod Γ ,根据矩阵乘法的定义得证。
表示为有理函数
对于上述数列 ( a j ) j ≥ 0 一定存在有理函数
Q ( x ) P ( x ) = j ≥ 0 ∑ a j x j
且 Q ( x ) = x d Γ ( x − 1 ) ,deg P < d 。我们称其为「有理函数 」是因为 P ( x ) , Q ( x ) 是「多项式 」。
证明 :对于 P ( x ) = ∑ j = 0 d − 1 p j x j 和 Q ( x ) := ∑ j = 0 d q j x j 考虑 Q ( x ) P ( x ) = ∑ j ≥ 0 q ~ j x j 的系数定义,这几乎就是形式幂级数「除法 」的定义,
q ~ N = ⎩ ⎨ ⎧ p 0 q 0 − 1 , ( p N − ∑ j = 1 N q j q ~ N − j ) ⋅ q 0 − 1 , − q 0 − 1 ∑ j = 1 d q j q ~ N − j , if N = 0 , else if N < d , otherwise .
我们只需要令
P ( x ) = ( ( j ≥ 0 ∑ a j x j ) ⋅ x d Γ ( x − 1 ) ) mod x d
那么根据 q ~ N 的定义,必然有 Q ( x ) P ( x ) = ∑ j ≥ 0 a j x j 。
Bostan–Mori 算法
计算单项
我们的目标仍然是给出上述多项式 P ( x ) , Q ( x ) ,求算 [ x k ] Q ( x ) P ( x ) 。
Bostan–Mori 算法基于 Graeffe 迭代,对于上述多项式 P ( x ) , Q ( x ) 有
Q ( x ) P ( x ) = Q ( x ) Q ( − x ) P ( x ) Q ( − x ) = V ( x 2 ) U 0 ( x 2 ) + x U 1 ( x 2 )
因为分母 V ( x 2 ) 是偶函数,所以子问题只需考虑其中的一侧
[ x k ] Q ( x ) P ( x ) = [ x ⌊ k /2 ⌋ ] V ( x ) U k mod 2 ( x )
我们付出两次多项式乘法的代价使得问题至少减少为原先的一半,而当 k = 0 时显然有 [ x 0 ] Q ( x ) P ( x ) = Q ( 0 ) P ( 0 ) ,时间复杂度同上。
计算连续若干项
目标是给出上述多项式 P ( x ) , Q ( x ) ,求算 [ x [ L , R ) ] Q ( x ) P ( x ) 。下面的计算中我们只需考虑对答案「有影响 」的系数,这是 Bostan–Mori 算法的关键。
我们不妨假设 deg P < deg Q ,否则我们也可以通过一次带余除法使问题回到这种情况。
我们先考虑更简单的问题:
[ x [ L , R ) ] Q ( x ) 1 = [ x [ L , R ) ] Q ( x ) Q ( − x ) 1 ⋅ Q ( − x )
我们需要求出 [ x [ L − d e g Q , R ) ] Q ( x ) Q ( − x ) 1 然后作一次乘法并取出 x L , … , x R − 1 的系数。令 V ( x 2 ) = Q ( x ) Q ( − x ) 那么我们只需求出
[ x [ ⌈ 2 L − d e g Q ⌉ , ⌈ 2 R ⌉ ) ] V ( x ) 1
就可以还原出 [ x [ L − d e g Q , R ) ] Q ( x ) Q ( − x ) 1 。进而我们只需求出 [ x [ L − d e g P , R ) ] Q ( x ) 1 再和 P ( x ) 作一次乘法即可求出 [ x [ L , R ) ] Q ( x ) P ( x ) 。
上面的算法虽然已经可以工作,但是每一次的递归的时间复杂度与 R − L 相关,我们希望能至少在递归求算时摆脱 R − L ,更具体的,我们先考虑求算 [ x [ L , L + d e g Q + 1 ) ] Q ( x ) 1 ,考虑
[ x [ L , L + d e g Q + 1 ) ] Q ( x ) 1 = [ x [ L , L + d e g Q + 1 ) ] Q ( x ) Q ( − x ) 1 ⋅ Q ( − x )
我们需要求出
[ x [ L − d e g Q , L + d e g Q + 1 ) ] Q ( x ) Q ( − x ) 1
那么对于 V ( x 2 ) = Q ( x ) Q ( − x ) 而言,我们只需求出
[ x [ ⌈( L − d e g Q ) /2 ⌉ , ⌈( L + d e g Q + 1 ) /2 ⌉ ) ] V ( x ) 1
这是因为
[ x k ] Q ( x ) Q ( − x ) 1 = ⎩ ⎨ ⎧ [ x k /2 ] V ( x ) 1 , 0 , if k ≡ 0 ( mod 2 ) , otherwise .
我们知道 L + deg Q 和 L − deg Q 的奇偶性是一样的,所以
⌈ 2 L + deg Q + 1 ⌉ − ⌈ 2 L − deg Q ⌉ = { deg Q + 1 , deg Q , if L + deg Q ≡ 0 ( mod 2 ) , otherwise .
这样我们可以写出伪代码
1 2 3 4 5 6 Algorithm Slice-Coefficients ( Q , L ) : Input : Q ( x ) ∈ C [ x ] , L ∈ Z . Output : [ x [ L , L + d e g Q + 1 ) ] Q ( x ) − 1 . if L ≤ 1 then return [ x [ L , L + d e g Q + 1 ) ] Q ( x ) − 1 Use other algorithm to compute Q ( x ) − 1 V ( x 2 ) ← Q ( x ) Q ( − x ) k ← ⌈ 2 L − d e g Q ⌉ ( t k , … , t k + d e g Q ) ← Slice-Coefficients ( V , k ) T ( x ) ← x ( L − d e g Q ) mod 2 ∑ j = 0 d e g Q t j + k x 2 j return [ x [ d e g Q , 2 d e g Q + 1 ) ] T ( x ) Q ( − x )
但是只有这个算法还不够,我们需要重新找到一个有理函数并求算更多系数。
找到新的有理函数表示
我们知道 Q ( x ) 本身和 Q ( x ) − 1 的一部分连续的系数比如 [ x [ L , L + d e g Q ) ] Q ( x ) − 1 和 L ≥ 0 ,我们希望求出 [ x [ L + d e g Q , L + 2 d e g Q ) ] Q ( x ) − 1 ,这等价于我们要求某个 P ( x ) 且 deg P < deg Q 使得 Q ( x ) P ( x ) 的前 deg Q 项与 [ x [ L , L + d e g Q ) ] Q ( x ) − 1 相同。简单来说:递推关系(有理函数的分母)是不变的,我们所做的只是更换初值(有理函数的分子)。
具体的,考虑
Q ( x ) P ( x ) = j ≥ 0 ∑ a j x j
我们现在希望将递推前进 n 项,那么就是
j ≥ n ∑ a j x j − n = Q ( x ) x n P ( x ) − Q ( x ) x n Q ( x ) ∑ j = 0 n − 1 a j x j
我们先用一次 Slice-Coefficients ( Q , L − deg P ) 计算出 [ x [ L − d e g P , L − d e g P + d e g Q + 1 ) ] Q ( x ) − 1 ,然后我们扩展合并出 [ x [ L − d e g P , L + d e g Q ) ] Q ( x ) − 1 ,再重新计算一个分子使得
Q ( x ) P ( x ) = j ≥ 0 ∑ ( [ x L + j ] Q ( x ) P ( x ) ) x j
最后我们使用形式幂级数的除法计算出 [ x [ 0 , R − L ) ] Q ( x ) P ( x ) ,时间为 O ( M ( d ) log L + M ( R − L )) 。
参考文献
Alin Bostan, Ryuhei Mori.A Simple and Fast Algorithm for Computing the N -th Term of a Linearly Recurrent Sequence .