分解质因数
引入
非平凡因数
。
考虑朴素算法,因数是成对分布的,
的所有因数可以被分成两块,即
和
。只需要把
里的数遍历一遍,再根据除法就可以找出至少两个因数了。这个方法的时间复杂度为
。
当
时,这个算法的运行时间我们是无法接受的,希望有更优秀的算法。一种想法是通过随机的方法,猜测一个数是不是
的因数,如果运气好可以在
的时间复杂度下求解答案,但是对于
的数据,成功猜测的概率是
, 期望猜测的次数是
。如果是在
里进行猜测,成功率会大一些。我们希望有方法来优化猜测。
朴素算法
筛法
处查阅更多打表的信息。
例题:
CF 1445C
Pollard Rho 算法
。那么在一年有
天的情况下,当房间中有
个人时,至少有两个人的生日相同的概率约为
。
考虑一个问题,设置一个数据
,在
里随机选取
个数(
时就是它自己),使它们之间有两个数的差值为
。当
时成功的概率是
,当
时成功的概率是
(考虑绝对值,
可以取
或
),随着
的增大,这个概率也会增大最后趋向于 1。
利用最大公约数求出一个约数
欧几里得算法
有
。
我们每过一段时间将这些差值进行
运算,设
,如果某一时刻得到
那么表示分解失败,退出并返回
本身。每隔
个数,计算是否满足
。此处取
,可以根据实际情况进行调节。
注意到在环上更容易分解出因数,我们可以先迭代一定的次数。
实现
|
ll Pollard_Rho(ll x) {
ll t = 0;
ll c = rand() % (x - 1) + 1;
// 加速算法,这一步可以省略
for (int i = 1; i < 1145; ++i) t = f(t, c, x);
ll s = t;
int step = 0, goal = 1;
ll val = 1;
for (goal = 1;; goal <<= 1, s = t, val = 1) {
for (step = 1; step <= goal; ++step) {
t = f(t, c, x);
val = val * abs(t - s) % x;
// 如果 val 为 0,退出重新分解
if (!val) return x;
if (step % 127 == 0) {
ll d = gcd(val, x);
if (d > 1) return d;
ll d = gcd(val, x);
if (d > 1) return d;
22
|
from random import randint
from math import gcd
def Pollard_Rho(x):
c = randint(1, x-1)
s = t = f(0, c, x)
goal = val = 1
while True:
for step in range(1, goal+1):
t = f(t, c, x)
val = val * abs(t - s) % x
if val == 0:
return x #如果 val 为 0,退出重新分解
if step % 127 == 0:
d = gcd(val, x)
if d > 1:
return d
d = gcd(val, x)
if d > 1:
return d
s = t
goal <<= 1
val = 1
|
例题:
P4718【模板】Pollard-Rho 算法
对于一个数
,用
Miller Rabin 算法
判断是否为素数,如果是就可以直接返回了,否则用 Pollard-Rho 算法找一个因子
,将
除去因子
。再递归分解
和
,用 Miller Rabin 判断是否出现质因子,并用 max_factor 更新就可以求出最大质因子了。由于这个题目的数据过于庞大,用 Floyd 判环的方法是不够的,这里采用倍增优化的方法。
实现
|
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
int t;
long long max_factor, n;
long long gcd(long long a, long long b) {
if (b == 0) return a;
return gcd(b, a % b);
long long quick_pow(long long x, long long p, long long mod) { // 快速幂
long long ans = 1;
while (p) {
if (p & 1) ans = (__int128)ans * x % mod;
x = (__int128)x * x % mod;
p >>= 1;
return ans;
bool Miller_Rabin(long long p) { // 判断素数
if (p < 2) return 0;
if (p == 2) return 1;
if (p == 3) return 1;
long long d = p - 1, r = 0;
while (!(d & 1)) ++r, d >>= 1; // 将d处理为奇数
for (long long k = 0; k < 10; ++k) {
long long a = rand() % (p - 2) + 2;
long long x = quick_pow(a, d, p);
if (x == 1 || x == p - 1) continue;
for (int i = 0; i < r - 1; ++i) {
x = (__int128)x * x % p;
if (x == p - 1) break;
if (x != p - 1) return 0;
return 1;
long long Pollard_Rho(long long x) {
long long s = 0, t = 0;
long long c = (long long)rand() % (x - 1) + 1;
int step = 0, goal = 1;
long long val = 1;
for (goal = 1;; goal *= 2, s = t, val = 1) { // 倍增优化
for (step = 1; step <= goal; ++step) {
t = ((__int128)t * t + c) % x;
val = (__int128)val * abs(t - s) % x;
if ((step % 127) == 0) {
long long d = gcd(val, x);
if (d > 1) return d;
long long d = gcd(val, x);
if (d > 1) return d;
void fac(long long x) {
if (x <= max_factor || x < 2) return;
if (Miller_Rabin(x)) { // 如果x为质数
max_factor = max(max_factor, x); // 更新答案
return;
long long p = x;
while (p >= x) p = Pollard_Rho(x); // 使用该算法
while ((x % p) == 0) x /= p;
fac(x), fac(p); // 继续向下分解x和p
int main() {
scanf("%d", &t);
while (t--) {
srand((unsigned)time(NULL));
max_factor = 0;
scanf("%lld", &n);
fac(n);
if (max_factor == n) // 最大的质因数即自己
printf("Prime\n");
|