ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

Miller–Rabin 素性检验算法

2022-01-11 23:04:28  阅读:178  来源: 互联网

标签:算法 59 素性 int Miller 29 素数 Rabin mod


算法介绍

Miller-Rabin素数检验或Rabin-Miller素数检验是一种概率素数检验:一种确定给定数是否可能是素数的算法,类似于费马素数检验和Solovay-Strassen素数检验。作为实践中使用比较广泛的素性检验算法的一种, Miller-Rabin算法最早在1976年由Gary L. Miller提出(当时该算法是确定性的), 并在1980年由Michael O. Rabin改进为无条件的概率算法.

数学原理

算法的主要思想就是检验数是否满足特定的素数条件, 这些条件在Miller-Rabin算法中是一个强概率条件, 所以通过条件检验的数, 一般能以大概率认定为素数.

设检验数n > 2是一个奇数, 可以改写为 n = 2 s ∗ d + 1 n = 2^s * d + 1 n=2s∗d+1, 其中s, d均为正数, 且d为奇数.
引入一个基数 a , ( 0 < a < n ) a, (0 < a < n) a,(0<a<n), 当满足以下的同余式条件之一时, 称n为关于a的强概率素数
条件包括
(1) a d ≡ 1 (   m o d     n ) a^{d} \equiv 1 \quad(\bmod \ n) ad≡1(mod n)
(2) a 2 r ⋅ d ≡ − 1 (   m o d     n )  for some  0 ≤ r < s a^{2^{r} \cdot d} \equiv-1 \quad(\bmod\ n) \text { for some } 0 \leq r<s a2r⋅d≡−1(mod n) for some 0≤r<s

主要依据是
费马小定理
对于素数n, 和正数a, 且n不整除a, 有 a n − 1 ≡ 1 (   m o d     n ) a^{n-1} \equiv 1 \quad(\bmod \ n) an−1≡1(mod n)

基于1(mod n)的平方根只有1, -1的观察

算法示例
举几个例子方便直观感受
(1) n = 233
233 − 1 = 2 3 ∗ 29 233 - 1 = 2^3 * 29 233−1=23∗29, 这里的 s = 3 , d = 29 s = 3, d = 29 s=3,d=29, 选择base a = 2 a = 2 a=2
检验: a d   m o d   n = 2 29   m o d   233 = 1 a^d\ mod\ n = 2^{29}\ mod\ 233 = 1 ad mod n=229 mod 233=1, 所以判定为素数

(2) n = 237
237 − 1 = 2 2 ∗ 59 237 - 1 = 2^2 * 59 237−1=22∗59, 这里的 s = 2 , d = 59 s = 2, d = 59 s=2,d=59, 选择base a = 2 a = 2 a=2
检验: a d   m o d   n = 2 59   m o d   237 = 167 a^d\ mod\ n = 2^{59}\ mod\ 237 = 167 ad mod n=259 mod 237=167, 所以判定为合数(非素数

源码实现

// C++ program Miller-Rabin primality test
#include <bits/stdc++.h>
using namespace std;

// Utility function to do modular exponentiation.
// It returns (x^y) % p
int power(int x, unsigned int y, int p)
{
	int res = 1;	 // Initialize result
	x = x % p; // Update x if it is more than or
				// equal to p
	while (y > 0)
	{
		// If y is odd, multiply x with result
		if (y & 1)
			res = (res*x) % p;

		// y must be even now
		y = y>>1; // y = y/2
		x = (x*x) % p;
	}
	return res;
}

// This function is called for all k trials. It returns
// false if n is composite and returns true if n is
// probably prime.
// d is an odd number such that d*2<sup>r</sup> = n-1
// for some r >= 1
bool miillerTest(int d, int n)
{
	// Pick a random number in [2..n-2]
	// Corner cases make sure that n > 4
	int a = 2 + rand() % (n - 4);

	// Compute a^d % n
	int x = power(a, d, n);

	if (x == 1 || x == n-1)
	return true;

	// Keep squaring x while one of the following doesn't
	// happen
	// (i) d does not reach n-1
	// (ii) (x^2) % n is not 1
	// (iii) (x^2) % n is not n-1
	while (d != n-1)
	{
		x = (x * x) % n;
		d *= 2;

		if (x == 1)	 return false;
		if (x == n-1) return true;
	}

	// Return composite
	return false;
}

// It returns false if n is composite and returns true if n
// is probably prime. k is an input parameter that determines
// accuracy level. Higher value of k indicates more accuracy.
bool isPrime(int n, int k)
{
	// Corner cases
	if (n <= 1 || n == 4) return false;
	if (n <= 3) return true;

	// Find r such that n = 2^d * r + 1 for some r >= 1
	int d = n - 1;
	while (d % 2 == 0)
		d /= 2;

	// Iterate given nber of 'k' times
	for (int i = 0; i < k; i++)
		if (!miillerTest(d, n))
			return false;

	return true;
}

// Driver program
int main()
{
	int k = 4; // Number of iterations

	cout << "All primes smaller than 100: \n";
	for (int n = 1; n < 100; n++)
	if (isPrime(n, k))
		cout << n << " ";

	return 0;
}

参考
[1] https://en.wikipedia.org/wiki/Miller%E2%80%93Rabin_primality_test
[2] https://www.geeksforgeeks.org/primality-test-set-3-miller-rabin/

标签:算法,59,素性,int,Miller,29,素数,Rabin,mod
来源: https://blog.csdn.net/qq_33976344/article/details/122442928

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有