This blog post is only available in Simplified Chinse.
在之前我们已经知道乘法逆元的三种求法,对于一般的题目让你把答案模一个质数,如果要求逆元一般用费马小定理,可以在 Θ(Nlog2(N)) 时间内构造出 1 到 N 的逆元:inv(x)=xmo−2modmo。但是对于 107 级别的 N,这样的求法就显得有点慢。能不能在 Θ(N) 时间内递推出 inv(x) 呢?
答案是肯定的。
线性推逆元的递推式
(模数是 N,⌊x⌋ 表示 x 向下取整)
inv(i)=(N−N/i)∗inv(Nmodi)modN
原理与证明
在某位大佬的博客上看到一个证明。(以下 a/b 与 ⌊ba⌋ 均表示 a 整除 b )
假设 k=⌊iN⌋,b=Nmodi;显然有:
k∗i+b≡0(modN)
变换得到:
−k∗i≡b(modN)
两边同除 i∗k:
−k∗inv(b)≡inv(i)(modN)
把 k 和 b 换回来,就得到了!
−⌊iN⌋∗inv(Nmodi)≡inv(i)(modN)inv(i)modN=−(N/i)∗inv(Nmodi)modN
因为 inv(i) 有很多个,我们可以求出最小的一个,一定小于等于 N(证明详见:乘法逆元的三种求法),所以左边的 modN 可以去掉了。右边有个 modN ,为了让右边大于 0,我们可以给它加上 N。
inv(i)=(N−N/i)∗inv(Nmodi)modN
初始 inv(0)=1,可以愉快地递推啦~
代码
其实知道了递推式,代码就两行……
inv[0]=1;
for (int i=1;i<=M;i++) inv[i]=(N-(N/i))*inv[N%i]%N;
参考
逆元详解 - CSDN博客