LogoCSP Wiki By Yundou
0 MathBasics

同余方程

同余符号

形如" \equiv " 的符号称作同余符号,对于以下式子:

ab(modn)a\equiv b\pmod n

其意义为:aa除以nn的余数等于bb除以nn的余数。

同余方程定义

形如

axb(modn)ax\equiv b\pmod n

的方程称为 线性同余方程(Linear Congruence Equation)。其中,aabbnn 为给定整数,xx 为未知数。需要从区间 [0,n1][0, n-1] 中求解 xx,当解不唯一时需要求出全体解。

用逆元求解

首先考虑简单的情况,当 aann 互素(coprime 或 relatively prime)时,即 gcd(a,n)=1\gcd(a, n) = 1

此时可以计算 aa 的逆元,并将方程的两边乘以 aa 的逆元,可以得到唯一解。

证明

xba1(modn)x\equiv ba ^ {- 1} \pmod n

接下来考虑 aann 不互素(not coprime),即 gcd(a,n)1\gcd(a, n) \ne 1 的情况。此时不一定有解。例如,2x1(mod4)2x\equiv 1\pmod 4 没有解。

g=gcd(a,n)g = \gcd(a, n),即 aann 的最大公约数,其中 aann 在本例中大于 1。

bb 不能被 gg 整除时无解。此时,对于任意的 xx,方程 axb(modn)ax\equiv b\pmod n 的左侧始终可被 gg 整除,而右侧不可被 gg 整除,因此无解。

如果 gg 整除 bb,则通过将方程两边 aabbnn 除以 gg,得到一个新的方程:

axb(modn)a^{'}x\equiv b^{'} \pmod{n^{'}}

其中 aa^{'}nn^{'} 已经互素,这种情形已经解决,于是得到 xx^{'} 作为 xx 的解。

很明显,xx^{'} 也将是原始方程的解。这不是唯一的解。可以看出,原始方程有如下 gg 个解:

xi(x+in)(modn)for i=0g1x_i\equiv (x^{'} + i\cdot n^{'}) \pmod n \quad \text{for } i = 0 \ldots g-1

总之,线性同余方程的 解的数量 等于 g=gcd(a,n)g = \gcd(a, n) 或等于 00

用扩展欧几里得算法求解

根据以下两个定理,可以求出线性同余方程 axb(modn)ax\equiv b \pmod n 的解。

定理 1:线性同余方程 axb(modn)ax\equiv b \pmod n 可以改写为如下线性不定方程:

ax+nk=bax + nk = b

其中 xxkk 是未知数。这两个方程是等价的,有整数解的充要条件为 gcd(a,n)b\gcd(a,n) \mid b

应用扩展欧几里德算法可以求解该线性不定方程。根据定理 1,对于线性不定方程 ax+nk=bax+nk=b,可以先用扩展欧几里得算法求出一组 x0,k0x_0,k_0,也就是 ax0+nk0=gcd(a,n)ax_0+nk_0=\gcd(a,n),然后两边同时除以 gcd(a,n)\gcd(a,n),再乘 bb。就得到了方程

abgcd(a,n)x0+nbgcd(a,n)k0=ba\dfrac{b}{\gcd(a,n)}x_0+n\dfrac{b}{\gcd(a,n)}k_0=b

于是找到方程的一个解。

定理 2:若 gcd(a,n)=1\gcd(a,n)=1,且 x0x_0k0k_0 为方程 ax+nk=bax+nk=b 的一组解,则该方程的任意解可表示为:

x=x0+ntx=x_0+nt k=k0atk=k_0-at

并且对任意整数 tt 都成立。

根据定理 2,可以从已求出的一个解,求出方程的所有解。实际问题中,往往要求出一个最小整数解,也就是一个特解

x=(xmodt+t)modtx=(x \bmod t+t) \bmod t

其中有

t=ngcd(a,n)t=\dfrac{n}{\gcd(a,n)}

如果仔细考虑,用扩展欧几里得算法求解与用逆元求解,两种方法是等价的。

实现

int ex_gcd(int a, int b, int& x, int& y) {
  if (b == 0) {
    x = 1;
    y = 0;
    return a;
  }
  int d = ex_gcd(b, a % b, x, y);
  int temp = x;
  x = y;
  y = temp - a / b * y;
  return d;
}
 
bool liEu(int a, int b, int c, int& x, int& y) {
  int d = ex_gcd(a, b, x, y);
  if (c % d != 0) return false;
  int k = c / d;
  x *= k;
  y *= k;
  return true;
}

On this page