模逆元(modular multiplicative inverse)俗称模倒数,是指与原数的乘积和 1 同余的整数。即对于整数 $a$ 和 $n$:

\[ax \equiv 1 \mod(n)\]

上式中 x 就是 a 对于 n 的模逆元,$a$ 与 $n$ 互质时模倒数存在,否则不存在。 它的数学意义是模倒数存在时,除以一个数和乘以它的模倒数同余,可以把除法转换为乘法。 这对于计算大数(10 以上)组合问题非常有效,适用于 Sherlock and PermutationsMatrix Tracing 等问题。 这类问题的都可以转化为组合问题 ${n \choose m}$,由于它的值通常会很大,我们只关心它对某个大质数 M(比如 $1e9 + 7$)的模:

\[{n \choose m} = C_n^m = \frac{n!}{m!(n-m)!} \mod{M}\]

计算思路

当 n 较大时(比如 1000)分子分母都会非常大,因此无法得到商后再取模。 一种办法是把分子分母都按照 1000 内的质数展开,再做矢量减法。 但是当 n 达到 1e6 或者更大时,质数的数量 ${n}/{\ln n}$ 逼近同样量级, 这么大的矢量计算也会不可行。 这时只能通过模倒数来把分母转换到分子:

\[\frac{b}{a} \equiv bx \mod(M)\]

其中 $x$ 是 $a$ 的模倒数,这样就可以把 ${n \choose m}$ 中的 $m!$ 和 $(n-m)!$ 换成对应的模倒数并移到分子上来,转换为复杂度为 $O(n)$ 的模乘法问题:

typdef unsigned long long LL;
const LL M = 1000000007;

int comb(LL n, LL m) {
    LL a = 1, b = 1;
    for(int i = n + m; i > m; i--) a = a * i % M;
    for(int i = n; i > 1; i--) b = b * i % M;
    int x = modInverse(b, M)    // 模倒数的计算,见下文
    return a * x % M;
}

本文接下来的部分只讨论如何计算 $a$ 的模倒数:

\[x = a^{-1} \mod(n)\]

暴力算法

暴力算法的思路是尝试每一个 x,直到找到了模逆元为止。复杂度是 $O(n)$:

LL modInverse(LL a, LL M) {
    a %= M;
    for (LL x = 2; x < M; x++) {
        if (x * a % M == 1) return x;
    }
    return -1;
}

它的复杂度是 $O(M)$,对于 $M = 1e9 + 7$ 来讲是无法接受的。 下面我们通过费马小定理和扩展欧几里得算法两个方向来降低复杂度。

费马小定理

费马小定理(Fermat's Little Theorem)指出,小于质数 p 的整数的 p 次方和它自己对 p 同余:

\[a^p \equiv a \mod(p)\]

我们再做一个转换,可以看到 $a$ 的模倒数可以通过指数运算得到:

\[x = a^{-1} \equiv a^{p - 2} \mod(p)\]

也就是说 1000 对于 $1e9 + 7$ 的模倒数为 $1000^{1e9 + 7}$。 暴力计算的复杂度是 $O(p)$,下面介绍的快速指数运算可以把复杂度降低到 $O(\lg p)$。

快速指数运算

快速指数运算(fast exponentiation)的思路非常简单,就是递归的二分法。 比如计算 $2^{64}$ 只需要先计算 $2^{32}$ 再对它进行平方。直接上代码:

LL expMod(LL a, LL p) {
    a %= M;
    if (p == 0) return 1;
    if (p & 1) return (a * expMod(a, p - 1)) % M;
    LL half = exp(a, p / 2);
    return half * half % M;
}

这样 $a$ 的模逆元 $x = a^{-1} \equiv a^{p - 2} \mod(p)$ 就是:

int modInverse(LL a, LL M) {
    return expMod(a, M - 2) % M;
}

它的复杂度是 $lg M$,这样组合算法的复杂度为 $O(n + lg M)$,前者 $O(n)$ 来自 comb() 函数里的阶乘。

欧拉定理

费马小定理成立的条件是模 n 为质数,如果 n 不是质数呢? 只要 n 和 a 互质(即模逆元存在),我们就可以使用费马小定理的泛化形式“欧拉定理”:

\[a^{\varphi(n)} = 1 \mod(n)\]

其中 $\varphi(n)$ 是欧拉函数表示小于 n 的数中与 n 互质的自然数的个数。 费马小定理是欧拉定理在 n 为质数时的特殊形式,这时 $\varphi(n) = n - 1$),代入得到:

\[a^{\varphi(n)} = a^{n - 1} = 1 \mod(n)\]

即费马小定理(令质数 $n=p$):

\[a^p = a \mod(p)\]

使用欧拉定理的关键在于计算欧拉函数 $\varphi(n)$,如果使用辗转相除法来判定,找到 n 以内的欧拉函数值复杂度为 $O(n * lgn)$。 对于典型的 $n = 1e9$ 的量级计算量太大,而且欧拉定理要求合数 n 和 a 互质, 所以欧拉定理通常并不实用(如果 n 为质数可以直接应用费马小定理)。

扩展欧几里得算法

欧几里得算法 俗称辗转相除法,是求最大公约数的常用算法,其复杂度为 $O(\lg n)$。 龙胆虎威中的四加仑水迷题(Die Hard 3) 一文中用到的也是该算法。 我们知道最大公约数总是可以写作两个数的线性组合,辗转相除法最终得到的正是最大公约数:

\[xa + yb = gcd(a, b)\]

扩展欧几里得算法 是在运行欧几里得算法的同时记录下系数 $x, y$。 这一系数就是贝祖数(Bézout's identity)。 在模逆元问题中 $a = a, b = n$,且 $a$ 与 $n$ 互质(这时才存在逆元),代入后得到:

\[xa + yn = gcd(a, n) = 1\] \[xa = 1 \mod(n)\]

即 a 对应的贝祖数就是 a 的模逆元。接下来的问题就是扩展欧几里得算法的实现:

int extEuclid (int a, int b, LL& x, LL& y) {
    if (a < b) return extEuclid(b, a, y, x);
    if (a % b == 0) {
        x = 0; y = 1;
        return b;
    }
    LL cx, cy;
    int ans = extEuclid(b, a % b, cx, cy);
    x = cy;
    y = (cx - (cy * (a / b  % M) % M) + 2 * M) % M;
    return ans;
}

对于本文涉及的组合问题 a 和 n 互质,GCD 总是 1,extEuclid() 的返回值并不重要。 我们只需要得到贝祖系数 x 的值,它就是 $a$ 的模逆元 $x=a^{-1}$:

int modInverse(LL a, LL M) {
    LL x, y;
    extEuclid(a, M, x, y);
    return x;
}

扩展欧几里得算法的复杂度是 M 的位数 $O(lg M)$,这样组合算法的复杂度为 $O(n + lg M)$,前者 $O(n)$ 来自 comb() 函数里的阶乘。

本文采用 知识共享署名 4.0 国际许可协议(CC-BY 4.0)进行许可,转载注明来源即可: https://harttle.land/2019/10/20/modular-multiplicative-inverse.html。如有疏漏、谬误、侵权请通过评论或 邮件 指出。