2025.07.26 答疑工作

[GESP202503 五级] 原根判断

小 A 知道,对于质数 \(p\) 而言,\(p\) 的原根 \(g\) 是满足以下条件的正整数:

  • \(1 < g < p\)
  • \(g^{p - 1} \bmod p = 1\)
  • \(\forall i, 1 \leq i < p - 1\) ,均有 \(g^{i} \bmod p \neq 1\)

小 A 现在有一个整数 \(a\),请你帮他判断 \(a\) 是不是 \(p\) 的原根。

做法:
如果学过原根,能不看题目给的条件,直接用比较好的方法做!\(b\) 的原根模 \(b\) 的阶,是 \(b\) 的欧拉函数。

根据阶的最小性扬掉 \(\varphi(b)\) 的多余因子求 \(a\)\(b\) 的阶 \(\delta_{b}(a)\) ,如果发现 \(\delta_{b}(a)\) 就是 \(\varphi(b)\) ,则 \(a\)\(b\) 的一个原根。

但是,通过充分挖掘题目给出的信息,只需要懂“整除、同余”,再配合数学归纳就能做!

显然我们知道 \(g^{0} \bmod p = 1\) 。题目的第 \(2\) 个条件又告诉了我们 \(g^{p - 1} \bmod p = 1\) ,那么说明在 \(\bmod p\) 意义下,\(0 = p-1\)
稍微分析一下前几项

\[\begin{aligned} g^{0} \bmod p &= g^{p-1} \bmod p \\ g^{1} \bmod p &= g^{0 + 1} = g^{p-1 + 1} = g^{p} \bmod p \\ g^{2} \bmod p &= g^{0 + 2} = g^{p-1 + 2} = g^{p+1} \bmod p \\ \end{aligned} \]

数学归纳后面的项数

\[\begin{aligned} g^{0 + x} \bmod p &= g^{p-1 + x} \bmod p \\ \end{aligned} \]

从上面的通项归纳出每 \(p - 1\) 个数是一个循环。

现在虽然知道了 \(p-1\) 是一个循环,但我们不保证 \(p-1\) 是最小的循环。若存在 \(i\) 满足 \(0 < i < p-1\)\(a^{i} \bmod p = 1\) ,那么 \(i\) 就是一个更小的循环( \(i\) 是循环证明方法和上面的归纳法一样)。

缺什么来什么,刚好现在题目的第 \(3\) 个条件又告诉我们 \(\forall i, 1 \leq i < p - 1\) ,均有 \(g^{i} \bmod p \neq 1\) ,那么说明不存这样的 \(i\)

于是 \(p-1\) 就是最小的循环。

根据整除性质,小的循环整除大的循环,即小的循环是大的循环的因子。

于是做法就是,我们先判断 \(p-1\) 是一个循环。然后分解 \(p-1\) 的因子,他的所有因子都不能是循环。

于是代码呼之欲出。实现的时候我们其实要知道怎么做快速幂,怎么做因子分解。

view
#include <bits/stdc++.h>
using namespace std;

typedef long long i64;

i64 ksm(i64 a, i64 n, int m) {
  i64 res = 1;
  for( ;n; n >>= 1, a = a * a % m) {
    if (n & 1) {
      res = res * a % m;
    }
  }
  return res;
}

int SOLVE() {
  int a, p;
  cin >> a >> p;
  
  assert(ksm(2, 3, p) == 8 % p);

  if (ksm(a, p - 1, p) != 1) {
    // 如果 p - 1 不是循环,坏了
    return -1;
  }

  for (int i = 2; i * i <= p - 1; i++) {
    if ((p - 1) / i * i == p - 1) {
      // 分解出 p - 1 的共轭因子 u 和 v
      int u = i, v = (p - 1) / u;
      if (ksm(a, u, p) == 1 || ksm(a, v, p) == 1) {
        // 如果 u 是循环,或者 v 是循环,说明 p - 1 不是最小循环,坏了
        return -1;
      }
    }
  }

  // 一切安好
  return 0;
}

signed main() {
  int T = 1;
  for (cin >> T; T--; ) {
    int ac = SOLVE();
    cout << (ac == 0 ? "Yes" : "No") << "\n";
  }
  return 0;
}

原根的存在性会更困难一些,但是大多数问题要么保证 \(m\) 的原根存在,要么询问某个数是不是 \(m\) 的阶,这些问题并不会涉及到原根存在性。

posted @ 2025-07-27 02:05  03Goose  阅读(45)  评论(0)    收藏  举报