java 素性测试_miller_rabin_素性测试

这篇博客详细解释了素性测试中的Miller-Rabin算法,该算法基于费马小定理和二次探测定理,提高了素数判定的效率。通过随机选择底数进行测试,避免将Carmichael数误判为素数。文章介绍了算法的原理和实现,并提供了代码示例,适合对数论和算法感兴趣的读者。
摘要由CSDN通过智能技术生成

看了好久没看懂,最后在这篇博客中看明白了。

费马定理的应用,加上二次探测定理。

Fermat素数测试

1819年有人发现了Fermat小定理逆命题的第一个反例:虽然2的340次方除以341余1,但341=11*31。后来,人们又发现了561, 645, 1105等数都表明a=2时Fermat小定理的逆命题不成立。人们把所有能整除2^(n-1)-1的合数n叫做伪素数(pseudoprime)。

不满足

0dd435c4469177d1867c5fa1f8394cff.png的n一定不是素数;如果满足的话则多半是素数。这样,一个比试除法效率更高的素性判断方法出现了:制作一张伪素数表,记录某个范围内的所有伪素数,那么所有满足

0dd435c4469177d1867c5fa1f8394cff.png且不在伪素数表中的n就是素数。之所以这种方法更快,是因为我们可以使用二分法快速计算

51a25437da903842d7067d71f34ec7f6.png的值(快速幂)。

然而不借助伪素数表的时候,算法出错的概率太高,需要改进.

我们刚才只考虑了a=2的情况。一个合数可能在a=2时通过了测试,但a=3时的计算结果却排除了素数的可能。于是,人们扩展了伪素数的定义,称满足a^(n-1) mod n = 1的合数n叫做以a为底的伪素数(pseudoprime to base a)。

随机选择若干个小于待测数的正整数作为底数a进行若干次测试,只要有一次没有通过测试就立即把这个数扔回合数的世界。这就是Fermat素性测试。

费马小定理毕竟只是素数判定的一个必要条件.满足费马小定理条件的整数n未必全是素数.有些合数也满足费马小定理的条件***.这些合数被称作Carmichael数,前3个Carmichael数是561,1105,1729. Carmichael数是非常少的.在1~100000000范围内的整数中,只有255个Carmichael数.

***费马小定理的前提是a和n互质。当n本身就是素数的时候如果a

利用下面的二次探测定理可以对上面的素数判定算法作进一步改进,以避免将Carmichael数当作素数.

Miller_Rabin素数测试算法

二次探测定理优化

Miller和Rabin两个人的工作让Fermat素性测试迈出了革命性的一步,建立了Miller-Rabin素性测试算法。新的测试基于下面的定理:

如果p是素数,x是小于p的正整数,且

4f81765ba9e5ebd83a04066851ff4928.png,那么要么x=1,要么x=p-1。

这是显然的,因为

4f81765ba9e5ebd83a04066851ff4928.png相当于p能整除

3f4791257fce5a9dd78760c5db726e73.png,也即p能整除(x+1)(x-1)。

由于p是素数,那么只可能是x-1能被p整除(此时x=1) 或 x+1能被p整除(此时x=p-1)。

我们下面来演示一下上面的定理如何应用在Fermat素性测试上。前面说过341可以通过以2为底的Fermat测试,因为2^340 mod 341=1。如果341真是素数的话,那么2^170mod 341只可能是1或340;当算得2^170 mod 341确实等于1时,我们可以继续查看2^85除以341的结果。我们发现,2^85 mod 341=32,这一结果摘掉了341头上的素数皇冠

这就是Miller-Rabin素性测试的方法。不断地提取指数n-1中的因子2,把n-1表示成

047437773b16cc1b87327867c544fc37.png(其中d是一个奇数)。那么我们需要计算的东西就变成了

6b2652b8ec58cdfbabc271821d1e740d.png除以n的余数。于是,

75102ca1afae4adacc7951cc8de17c92.png要么等于1,要么等于n-1。如果

75102ca1afae4adacc7951cc8de17c92.png等于1,定理继续适用于

c0675e65ce98f355989945926d85a460.png,这样不断开方开下去,直到对于某个i满足

6cb0df87606cd65b2a2d0fe8dda30aba.png或者最后指数中的2用完了得到的

4f3001fe30fde6435d66ac3ae7ee6bb6.png。这样,Fermat小定理加强为如下形式:

尽可能提取因子2,把n-1表示成

047437773b16cc1b87327867c544fc37.png,如果n是一个素数,那么或者

83c9050ed5199bc0b8ec13a4e6464b3a.png,或者存在某个i使得

6cb0df87606cd65b2a2d0fe8dda30aba.png ( 0<=i

5075ace10e670be5d8cccb0f50c49475.png的情况统一到后面去了)

Miller-Rabin素性测试同样是不确定算法,我们把可以通过以a为底的Miller-Rabin测试的合数称作以a为底的强伪素数(strong pseudoprime)。第一个以2为底的强伪素数为2047。第一个以2和3为底的强伪素数则大到1 373 653。

Miller-Rabin算法的代码也非常简单:计算d和r的值(可以用位运算加速,即快速积,快速幂),然后二分计算

b50dde7204c0f2736fd6d056177e02de.png的值,最后把它平方r次。

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

/*对应hoj 1356 Prime Judge*/#include

#define MT 5

using namespacestd;

typedeflong longll;int prime[] = {2, 3, 7, 61, 24251};

inline ll mulmod(ll a, ll b, ll k) {//*标出了核心语句//a %= k;//b %= k;

if (b < 0) {///将b变为正的

a = -a;

b= -b;

}

ll re= 0, temp =a;///re为最终结果,temp保持循环.re需要temp的时候,就加一下,否则temp继续累乘

while(b) {if (b & 1) re += temp;///二进制思想,需要即加*

//re %= k;

b >>= 1;///下一位等待检测**

temp <<= 1;///temp一直累乘***

//temp %= k;

}return re%k;*/

/*实际上呢,用以上的函数在hoj 1356是会TLE的(mod太多次了...)~应该用下面的方法...*/

return (a*b)%k;//-_-b

}//此时不需要再模,于是只剩下核心语句~快速幂和快速积都是二进制思想,核心是一样的

inline ll powermod(ll a, ll b, ll k) {

ll re= 1, temp =a;while(b) {if (b & 1) re = mulmod(re, temp, k);//只是把"加"入答案变为"乘"入答案

temp =mulmod(temp, temp, k);

b>>= 1;

}returnre;

}intTwiceDetect(ll a, ll b, ll k) {int t = 0;

ll x, y;while ((b & 1) == 0) {

b>>= 1;

t++;

}///b = d * 2^t; b = d;

y = x = powermod(a, b, k);///x = y = a^d % n///二次探测定理是反向递归的,当然也可以用如下的正向迭代法去测试

while (t--) {

y=mulmod(x, x, k);if (y == 1 && x != 1 && x != k - 1)///注意y!=1的时候是不做判断的,即对应

return 0;///递归时在某一环节x==p-1的情况,对此x开方则无意义,但是迭代的话不能break,只能ignore并继续.

x = y;///继续向高次迭代,那么至少最后一次应该是等于1的(Fermat小定理)

}returny;

}boolMiller_Rabin(ll n) {inti;

ll tmp;for (i = 0; i < MT; i++) {

tmp=prime[i];if (n == prime[i]) return true;if (TwiceDetect(tmp, n - 1, n) != 1)break;

}return (i ==MT);

}intmain() {

ll n;while (scanf("%lld", &n) == 1) {if ((n > 1) &&Miller_Rabin(n)) {

printf("YES\n");

}else{

printf("NO\n");

}

}return 0;

}

View Code

对于大数的素性判断,目前Miller-Rabin算法应用最广泛。一般底数仍然是随机选取,但当待测数不太大时,选择测试底数就有一些技巧了。比如,如果被测数小于4 759 123 141,那么只需要测试三个底数2, 7和61就足够了。当然,你测试的越多,正确的范围肯定也越大。如果你每次都用前7个素数(2, 3, 5, 7, 11, 13和17)进行测试,所有不超过341 550 071 728 320的数都是正确的。如果选用2, 3, 7, 61和24251作为底数,那么10^16内唯一的强伪素数为46 856 248 255 981。这样的一些结论使得Miller-Rabin算法在OI中非常实用。通常认为,Miller-Rabin素性测试的正确率可以令人接受,随机选取k个底数进行测试算法的失误率大概为4^(-k)。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值