跳转至

二次剩余

引入

二次剩余可以认为是在讨论求模意义下 开平方 运算的可行性。对于更高次方的开方可参见 k 次剩余

定义

二次剩余

令整数 ap 满足 (a,p)=1,若存在整数 x 使得

x2a(modp),

则称 a 为模 p 的二次剩余,否则称 a 为模 p 的二次非剩余。后文可能在模 p 显然的情况下简写成二次(非)剩余。

Euler 判别法

当模数为奇素数时,我们有如下定理:

Euler 判别法

对奇素数 p 和满足 (a,p)=1 的整数 a,有

ap12{1(modp),(xZ),  ax2(modp),1(modp),otherwise.

即对上述的 pa

  1. ap 的二次剩余当且仅当 ap121(modp).
  2. ap 的二次非剩余当且仅当 ap121(modp).
证明

首先由 Fermat 小定理ap11(modp),故

(ap12+1)(ap121)0(modp),

从而对任意满足 (a,p)=1a 均有 a(p1)/2±1(modp).

另外由 p 是奇素数,我们有:

xp1ap12=(x2)p12ap12=(x2a)P(x),

其中 P(x) 是某个整系数多项式,进而:

xpx=x(xp1ap12)+x(ap121)=(x2a)xP(x)+(ap121)x.

同余方程的定理 5 可知,a 是模 p 的二次剩余当且仅当 a(p1)/21(modp). 进而 a 是模 p 的非二次剩余当且仅当 a(p1)/21(modp).

基于 Euler 判别法,我们可以得到如下推论:

二次剩余的数量

对于奇素数 p,模 p 意义下二次剩余和二次非剩余均有 p12 个。

证明

根据 Euler 判别法,考虑 ap121(modp).

注意到 p12(p1),由 同余方程的定理 6 可知 ap121(modp)p12 个解。所以模 p 意义下二次剩余和二次非剩余均有 p12 个。

Legendre 符号

为了方便接下来的讨论,我们引入如下记号:

Legendre 符号

奇素数 p 和整数 a,定义 Legendre 符号如下:

(ap)={0,pa,1,(pa)((xZ),  ax2(modp)),1,otherwise.

即对于 (a,p)=1a

  • a 是模 p 的二次剩余当且仅当 (ap)=1.
  • a 是模 p 的二次非剩余当且仅当 (ap)=1.

下表为部分 Legendre 符号的值(From Wikipedia

性质

  1. 对任意整数 a

    ap12(ap)(modp).

    进一步,我们有推论:

    • (1p)=1.
    • (1p)=(1)p12={1,p1(mod4),1,p3(mod4).
  2. a1a2(modp)(a1p)=(a2p).

  3. 完全积性)对任意整数 a1,a2

    (a1a2p)=(a1p)(a2p).

    我们有推论:对整数 a,bpb

    (ab2p)=(ap).
  4. (2p)=(1)p218={1,p±1(mod8),1,p±3(mod8).
证明
  1. Legendre 符号的定义Euler 判别法 易得。
  2. 注意到

    a1a2(modp)(a1p)(a2p)(modp),

    |(a1p)(a2p)|2p>2,故:

    a1a2(modp)(a1p)=(a2p).
  3. 由 1 得

    (a1a2p)a1p12a2p12(a1p)(a2p)(modp).

    |(a1a2p)(a1p)(a2p)|2p>2,故

    (a1a2p)=(a1p)(a2p).
  4. 参见 二次互反律

基于如上性质,若对任意奇素数 pq(pq) 的值均可计算,则我们就可以对任意合法情况计算 Legendre 符号的值。接下来我们有一个优美的定理,这个定理巧妙地在 (pq)(qp) 之间建立起了联系,使得我们能用类似 辗转相除法 的思路完成计算。

二次互反律

二次互反律

pq 是两个不同的奇素数,则

(pq)(qp)=(1)p12q12.

证明方式很多1。一种证明方式是基于如下引理2

Gauss 引理

p 是奇素数,(n,p)=1,对整数 k (1k(p1)/2),令 rk=nkmodp,设 A={rk:rk<p/2}B={rk:rk>p/2},则

(np)=(1)|B|.
证明

λ=|A|μ=|B|,显然 λ+μ=(p1)/2,则

np12(p12)!=k=1p12nkaAabBb(modp).

我们知道对 B 中任意元素 b,有 p2<b<p,所以 0<pb<p2,进一步,对 B 中任意元素 b,我们有 pbA,否则若 A,B 中分别存在元素 a,b 使得 a=pb,则存在整数 0<k1,k2<(p1)/2 使得 a=nk1b=nk2pn(k1+k2),由于 (n,p)=1,则 p(k1+k2),注意到 0<k1+k2<p,所以产生矛盾。因此

np12(p12)!(1)μaAabB(pb)=(1)μ(p12)!(modp),

np12(1)μ(modp).

从而由 Legendre 符号的 性质 1 即得证。

推广

Gauss 引理可做如下推广7

p 是奇素数,令 IZp 满足 II=ZpII=,其中 I:={i:iI},则对任意与 p 互质的整数 n

(np)=(1)|J|,

其中 J={jI:njI}.

不难发现取 I={1,2,,(p1)/2} 即可得到 Gauss 引理。证明方法和 Gauss 引理的证明基本相同,故省略。

容易得到如下推论:

推论

对奇素数 p,有

(2p)=(1)p218={1,p±1(mod8),1,p±3(mod8).

对奇素数 p,奇数 n 满足 (n,p)=1,有

(np)=(1)i=1(p1)/2ni/p.
证明

对 Gauss 引理中的 n,k,rk,A,B,λ,μ,有 nk=pnkp+rk,进而

np218=k=1p12nk=pk=1p12nkp+aAa+bBb=pk=1p12nkp+aAa+bB(pb)+2bBbpμ=pk=1p12nkp+k=1p12k+2bBbpμ=pk=1p12nkp+p218+2bBbpμ,

因此

(n1)p218=pk=1p12nkp+2bBbpμ.

n=2,则 0<nkpp1p<1,从而有

p218μ(mod2).

2n,则有

k=1p12nkpμ(mod2).

根据如上推论,证明二次互反律只需验证

p12q12=k=1p12qkp+k=1q12pkq.

考虑由点 (px,qy)1xq12,1yp12 构成的集合 S,将其根据 pxqy 的大小关系分成两部分(显然 pxqy),分别验证三个集合的大小即可。

二次互反律不仅能用于判断数 n 是否是模 p 的二次剩余,还能用于确定使数 n 为二次剩余的模数的结构。

Example
  • 使得 5 为二次剩余的奇素数 p 满足 p±1(mod5).
  • 使得 3 为二次剩余的奇素数 p 满足 p1(mod3).
  • 使得 23 同时为二次剩余的奇素数 p 满足 p11(mod24).

另外,我们还可以证明诸如「形如 4k+1 的素数有无限多个」之类的结论,这一类结论实际上是 Dirichlet 定理 的简单推论。

Jacobi 符号

根据二次互反律,我们可以很自然地想到一种推广 Legendre 符号的方法:

Jacobi 符号

正奇数 m=p1α1pkαk 和整数 a,其中 p1,,pk 是素数,α1,,αk 是正整数,定义 Jacobi 符号如下:

(am):=i=1k(api)αi.

其中等式右端的 (api)Legendre 符号。另外对整数 a(a1)=1.

Warning

我们一般不区分 Legendre 符号和 Jacobi 符号,因为由完全积性可知 Jacobi 符号具有和 Legendre 符号一样的性质,所以这两种符号的计算方法是一致的。但是有一点需要注意:当 m 不是奇素数 时,(am) 的值与 a 是否是模 m 的二次剩余 无关,但是若 (am)=1,则说明 m 至少存在一个(实际上是奇数个)素因子 p 使得 a 是模 p 的二次非剩余,从而此时 a 是模 m 的二次非剩余。

我们还可以把模数进一步推广为 整数(只需补充 (a1)(a0)(a2) 的定义),这样就得到了 Kronecker 符号

相关算法

特殊情况时的算法

对于同余方程 x2a(modp),其中 p 为奇素数且 a 为二次剩余在 pmod4=3 时有更简单的解法,考虑

(a(p+1)/4)2a(p+1)/2(modp)xp+1(modp)(x2)(xp1)(modp)x2(modp)(Fermat's little theorem)

那么 a(p+1)/4modp 为一个解。

Atkin 算法

仍然考虑上述同余方程,此时 pmod8=5,那么令 b(2a)(p5)/8(modp)i2ab2(modp) 那么此时 i21(modp)ab(i1)modp 为一个解。

证明 i2(2ab2)2(modp)(2a(2a)(p5)/4)2(modp)((2a)(p1)/4)2(modp)(2a)p12(modp)1(modp)

那么

(ab(i1))2a2(2a)(p5)/4(2i)(modp)a(i)(2a)(p1)/4(modp)a(modp)

Cipolla 算法

Cipolla 算法用于求解同余方程 y2a(modp),其中 p 为奇素数且 a 为二次剩余。

本节考虑 Fp[x]/(x2g) 中的运算,其中 gFp

计算方法

不熟悉 多项式环 的读者,可以简单理解为该集合的元素都具有形式 a0+a1xa0,a1Fp,且遵循如下运算法则:

(a0+a1x)+(b0+b1x)(a0+b0)+(a1+b1)x(mod(x2g))(a0+a1x)(b0+b1x)(a0b0+a1b1g)+(a1b0+a0b1)x(mod(x2g))

需要注意的是,此处的 x 并不是一个具体的数,而是表示多项式中的形式记号。运算中一个关键的点在于利用 x2g(mod(x2g)) 将二次项转化为一次项和常数项。另外,所有整数运算都需要对 p 取模。

关于该结构的更多内容,请参见 多项式域论 等页面。

该算法的第一步为找到一个 r 使得 r2a 为二次非剩余。当然对于 a0(modp) 不可能找到这样的 r,需要进行特判。下文只讨论 a0(modp) 的情况。此时可随机一个 r 然后判断,期望可以 2 步找到。于是,(rx)p+12mod(x2(r2a)) 为一个解,可以通过快速幂求值。

为什么期望只需要两步

考虑 r2a 为二次剩余的情况,则存在一个 x 使得 r2ax2(modp),移项可得 (r+x)(rx)a(modp),不难发现每一个 (r+x)[1,p1],都一一对应于一组 (r,x) 的解,所以使原方程成立的解一共有 p1 组。我们分类讨论 x0x0 两种情况。对于 x0,由于 a 是二次剩余,对应了 2r 的取值;对于 x0,有 p12 种情况,每一个 r 对应其中两种,一共有 p32r 的取值。综上,一共有 2+p32=p+12 种情况使得 r2a 为二次剩余,所以每随机一次得到二次非剩余的概率就是 p12p,期望步数为 2pp12

证明

为了方便,首先令 f(x)=x2(r2a)Fp[x]

需要证明的是,(rx)p+12modf(x) 是原式的解,并且它属于 Fp。首先考虑证明前者,即证明 (rx)p+1a(modf(x))。为此,我们需要先证明两个引理:

引理 1: xpx(modf(x))

证明:

xp=x(x2)p12x(r2a)p12(modf(x))(x2r2a(modf(x)))x(modf(x))(r2a is quadratic non-residue)

引理 2:(a+b)pap+bp(modp)

使用二项式定理容易发现,除了第一项和最后一项,分子上的 p 无法消掉,于是只剩下 ap+bp

(a+b)p=i=0p(pi)aibpi=i=0pp!i!(pi)!aibpiap+bp(modp)

有了这两个引理,我们再来考虑证明原式:

(rx)p+1=(rx)p(rx)(rpxp)(rx)(modf(x))(r+x)(rx)(modf(x))=r2x2r2(r2a)(modf(x))=a

下面通过反证法证明我们求出的解属于 Fp,即其 x 的系数为 0

假设存在一个 (a0+a1x)2a(modf(x)) 满足 a10(modp),即 a02+2a0a1x+a12x2a(modf(x)),移项并化简可得:

a02+a12(r2a)a2a0a1x(modf(x))

式子左边的 x 的系数为 0,所以右边 x 的系数也为 0,即 a0a10(modp),由于我们令 a10(modp),所以一定有 a00(modp),于是 (a1x)2a(modf(x))r2aaa12(modp)

由于 aa12 都是二次剩余,由 Legendre 符号的积性可知 aa12 也是二次剩余,这与 r2a 是二次非剩余矛盾。于是原式不存在一个解使得 x 的系数非 0,我们求出的解的 x 的系数也必定为 0

模板题 洛谷 P5491【模板】二次剩余
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
#include <iostream>
#include <random>

long long p, v;  // 分别是模数和 r^2 - a 的值

struct Poly {
  long long a, b;

  Poly(long long _a = 0, long long _b = 0) : a(_a), b(_b) {}
};

Poly operator*(const Poly& x, const Poly& y) {
  // 重载乘法,可以参考上面有关运算性质的说明
  return Poly((x.a * y.a + v * (x.b * y.b % p)) % p,
              (x.a * y.b + x.b * y.a) % p);
}

// 多项式的快速幂,用于计算答案
Poly modpow(Poly a, long long b) {
  Poly res(1, 0);
  while (b) {
    if (b & 1) res = res * a;
    a = a * a;
    b >>= 1;
  }
  return res;
}

// 普通的快速幂,用于判断二次非剩余
long long modpow(long long a, long long b) {
  long long res = 1;
  while (b) {
    if (b & 1) res = res * a % p;
    a = a * a % p;
    b >>= 1;
  }
  return res;
}

// 用于生成随机数
std::mt19937 rng(std::random_device{}());

long long cipolla(long long a, long long _p) {
  p = _p;
  if (a == 0)
    return 0;  // 特判一下 0 的情况
  else if (modpow(a, (p - 1) / 2) == p - 1)
    return -1;  // 判断二次非剩余,此时无解
  else {
    // 随机 r,使得 r^2 - a 是一个二次非剩余
    long long r;
    for (r = rng() % p;; r = rng() % p) {
      if (modpow((r * r - a + p) % p, (p - 1) / 2) == p - 1) break;
    }
    v = (r * r - a + p) % p;
    return modpow(Poly(r, 1), (p + 1) / 2).a;  // 根据结论式计算结果
  }
}

int main() {
  int t, a, p;
  std::cin >> t;
  while (t--) {
    std::cin >> a >> p;
    int ans = cipolla(a, p);
    if (ans == -1)
      std::cout << "Hola!" << std::endl;
    else if (ans == 0)
      std::cout << 0 << std::endl;
    else {
      // 相反数是另一个解
      int ans2 = (p - ans) % p;
      if (ans2 < ans) std::swap(ans, ans2);
      std::cout << ans << " " << ans2 << std::endl;
    }
  }
  return 0;
}

Bostan–Mori 算法

该算法基于 Cipolla 算法,我们将问题转换为 常系数齐次线性递推 再应用 Bostan–Mori 算法。考虑另一种常见的 Cipolla 算法的描述:b=x(p+1)/2mod(x2tx+a) 为满足 b2a(modp) 的一个解3,其中 x2tx+aFp[x] 为不可约多项式。系数 t 的选取同样使用随机方法。证明过程略。参考 Bostan 和 Mori 论文4中的算法我们可以发现问题可转化为求解形式幂级数的乘法逆元的某一项系数:

b=[x(p+1)/2]11tx+ax2

[xn]k0+k1x1+k2x+k3x2={[x(n1)/2]k1k0k2+k1k3x1+(2k3k22)x+k32x2,if nmod2=1[xn/2]k0+(k0k3k1k2)x1+(2k3k22)x+k32x2,else if n0

n=0 时显然有 [x0]k0+k1x1+k2x+k3x2=k0,该算法乘法次数相较于 Cipolla 算法更少。其他相关乘法次数较少的算法可见 Müller 的文章5

Legendre 算法

对于同余方程 x2a(modp),其中 p 为奇素数且 a 为二次剩余。Legendre 算法可描述为找到 r 满足 r2a 为二次非剩余,令 a0+a1x=(rx)p12mod(x2a),那么 a00(modp)a12a(modp).

证明

考虑选择一个 b 满足 b2a(modp),那么 (rb)(r+b)=r2a 为二次非剩余,所以

(rb)p12(r+b)p121(modp)

存在环态射

ϕ:Fp[x]/(x2a)Fp×Fpx(b,b)

那么

(a0+a1b,a0a1b)=ϕ(a0+a1x)=ϕ(rx)p12=((rb)p12,(r+b)p12)=(±1,1)

所以 2a0=(±1)+(1)=02a1b=(±1)(1)=±2.

Tonelli–Shanks 算法

Tonelli–Shanks 算法是基于离散对数求解同余方程 x2a(modp) 的算法6,其中 p 为奇素数且 a 为模 p 的二次剩余。

p1=m2n,其中 m 为奇数。仍然使用随机方法寻找 rFp 满足 r 为二次非剩余。令 grm(modp)ba(m1)/2(modp),那么存在整数 e{0,1,2,,2n1} 满足 ab2ge(modp)。若 a 为二次剩余,那么 e 为偶数且 (abge/2)2a(modp).

证明

根据费马小定理可知

g2nrm2n=rp11(modp).

又由于 r 是二次非剩余,有

g2n1rm2n1=rp121(modp).

所以,gp 的阶是 2n。又因为 ab2am(modp)x2n1(modp) 的解,所以 amg 的幂次。记 amge(modp)

因为 a 是二次剩余,所以

ge2n1am2n1=ap121(modp).

由阶的性质可知,2ne2n1,所以,e 是偶数。因此,abge/2modp 是良定义的,且

(abge/2)2=a2b2geam+1gea(modp).

剩下的问题是如何计算 e。Tonelli 和 Shanks 提出一次确定 e 的一个二进制位。令 e 在二进制下表示为 e=e0+2e1+4e2+,其中 ek{0,1}。因为 a 是二次剩余,所以开始时 e0=0,然后利用如下性质逐位确定 ek 的取值:

(geg(emod2k))2n1kg2n1ek{1(modp),if ek=01(modp),if ek=1

其中,geab2(modp) 已知,而 emod2k 的取值可以由之前的数位 e0,e1,,ek1 计算得到。当然,实现算法时,只需要直接维护乘积 geg(emod2k)modp 即可。

习题

参考资料与注释

  1. Quadratic residue - Wikipedia
  2. Euler's criterion - Wikipedia

  1. Proofs of quadratic reciprocity - Wikipedia 

  2. Carl Friedrich Gauss. Untersuchungen über höhere Arithmetik, 1965. Page 458-462. 

  3. A. Menezes, P. van Oorschot and S. Vanstone. Handbook of Applied Cryptography, 1996. 

  4. Alin Bostan, Ryuhei Mori. A Simple and Fast Algorithm for Computing the N-th Term of a Linearly Recurrent Sequence. Available at https://arxiv.org/abs/2008.08822

  5. S. Müller, On the computation of square roots in finite fields, Design, Codes and Cryptography, Vol.31, pp. 301-312, 2004. 

  6. Daniel. J. Bernstein. Faster Square Roots in Annoying Finite Fields. 

  7. Kobi Kremnizer.Lectures in number theory 2022. Proposition 4.3.