斐波那契公约数
题目描述
对于Febonacci数列,求第n项和第m项的最大公约数是多少?(n,m<=1e9>)。对于最后的结果只要输出最后的8位数字就可以了。
思路
首先注意到的是数据量巨大(1e9),数组开不下;即使使用Febonacci递推公式也会超时。不过好在Febonacci数列是有通项公式的。因为$F[n] = F[n-2] + F[n-1]$,解特征方程$X^{2}=X+1$得到解$X=(1 \pm \sqrt{5})/2$,再代入$F[n]=aX_{1}^{n} + bX_{2}^{n}$,就可以解的最终的通项公式是$F[n] = \frac{1}{\sqrt{5}} × [(\frac{1+\sqrt{5}}{2})^n-(\frac{1-\sqrt{5}}{2})^n]$。
求两个数的最大公约数可以用 辗转相除法
1 | typedef long long LL; |
不过这里的数字会非常巨大,并且$\sqrt{5}$也不是一个整数,用快速幂的时候对double做%
运算似乎会报错……
题解
正解用到了 矩阵加速 和 __一个Febonacci的结论__。
矩阵加速并不是必须的,但是能减少不少计算量。先看看矩阵加速的巧妙之处:
矩阵加速
如果已知:$(F[2], F[1])^T$,那么要求$(F[3], F[2])^T$,只需要乘一个矩阵:
$$
\left [
\begin{matrix}
1 & 1 \
1 & 0 \
\end{matrix}
\right ]
$$
其实这里已经能看出用矩阵的方式表达Febonacci数列的递推关系。
如果初始化矩阵为 $(F[2], F[1])^T$,那么通项公式为:
$$
(F[n], F[n-1])^T = \left [
\begin{matrix}
1 & 1 \
1 & 0 \
\end{matrix}
\right ]^{n-2} (F[2], F[1])^T (n \ge 2)
$$
这样做就能使用 矩阵快速幂 减小复杂度到 O(logn)
算得 F[n]
。
再看一下这道题的 关键之处
一个Febonacci的结论
先看结论:
$$gcd(F[n], F[m]) = F[gcd(n, m)]$$
这个结论似乎很好记,但是不是很直接,Febonacci数列的第n项和第m项的最大公约数居然正好等于第$gcd(n,m)$项的值?
详细的推导过程是这样的:
- 如果$F[n]=a, F[n+1]=b$
- 那么$F[n+2]=a+b$, $F[n+3] = a+2b$, … ,$F[m] = F[m−n−1]a + F[m−n]b$
- 所以有$F[m] = F[m−n−1] ∗ F[n] + F[m−n] ∗ F[n+1]$
- 这样,$gcd(F[n], F[m]) = gcd(F[n]$, $F[m−n−1] ∗ F[n] + F[m−n] ∗ F[n+1])$
- 其中$F[m−n−1] ∗ F[n]$是$F[n]$的倍数,因此上式转为$gcd(F[n], F[m]) = gcd(F[n], F[m−n] ∗ F[n+1])$
- 可以通过递推证明:$gcd(F[n], F[n+1]) = 1$
- 因此$gcd(F[n], F[m]) = gcd(F[n], F[m−n])$,即$gcd(F[n], F[m]) = gcd(F[n], F[m%n])$
这样求解的目标就化简为求第$gcd(n,m)$项了。
代码
1 |
|