1. 程式人生 > >素數測試

素數測試

相同 mil != ++ uri can www. clu .com

作者:jostree 轉載請註明出處 https://www.cnblogs.com/jostree/p/10274890.html

費馬小定理

如果p是素數,a是小於p的正整數,那麽a^(p-1) mod p = 1。

首先我們證明這樣一個結論:如果p是一個素數的話,那麽對任意一個小於p的正整數a,a, 2a, 3a, ..., (p-1)a除以p的余數正好是一個1到p-1的排列。例如,5是素數,3, 6, 9, 12除以5的余數分別為3, 1, 4, 2,正好就是1到4這四個數。

反證法,假如結論不成立的話,那麽就是說有兩個小於p的正整數m和n使得na和ma除以p的余數相同。不妨假設n>m,則p可以整除a(n-m)。但p是素數,那麽a和n-m中至少有一個含有因子p。這顯然是不可能的,因為a和n-m都比p小。

用同余式表述,我們證明了: (p-1)! ≡ a * 2a * 3a * … * (p-1)a (mod p)
也即: (p-1)! ≡ (p-1)! * a^(p-1) (mod p)
兩邊同時除以(p-1)!,就得到了我們的最終結論: 1 ≡ a^(p-1) (mod p)

Miller-Rabbin 素數測試

如果p是素數,x是小於p的正整數,且x^2 mod p = 1,那麽要麽x=1,要麽x=p-1。這是顯然的,因為x^2 mod p = 1相當於p能整除x^2-1,也即p能整除(x+1)(x-1)。由於p是素數,那麽只可能是x-1能被p整除(此時x=1)或x+1能被p整除(此時x=p-1)。

從而我們選取不同的x,對p其進行進行素數測試,即可。代碼如下:

#include <iostream>
#include <string>
#include <cstring>
#include <cstdlib>
#include  <ctime>

using namespace std;
typedef long long LL;
const int S = 100;

LL mul(LL a, LL b, LL n)
{
    LL res = 0;
    while( b )
    {
        if( b & 1 )
        {
            res += a;
            res %= n;
        }
        a = (a + a) % n;
        b = b >> 1;
    }
    return res;
}

LL pow(LL a, LL b, LL n)
{
    LL res = 1;
    while( b )
    {
        if( b & 1 )
        {
            res = mul(res, a, n);
        }
        a = mul(a, a, n);
        b = b >> 1;
    }
    return res;
}

bool miller_rabbin(LL n)
{
    if( n == 2 ) return true;
    if(n < 2 || !(n & 1)) return false;
    int t = 0;
    LL a, x, y, u = n - 1;
    while((u & 1) == 0) t++, u>>=1;
    for( int i = 0 ; i < S ; i++ )
    {
        a = rand() % (n - 1) + 1;
        x = pow(a, u, n);
        for( int j = 0 ; j < t ; j++ )
        {
            y = mul(x, x, n);
            if( y == 1 && x != 1 && x != n - 1 )
            {
                return false;
            }
            x = y;
        }
        if( x != 1 ) return false;
    }
    return true;
}


int main(int argc, char *argv[])
{
    int T;
    scanf("%d", &T);
    LL N;
    srand(time(0));
    while( T-- )
    {
        cin>>N;
        if( miller_rabbin(N) )
        {
            cout<<"Yes"<<endl;
        }
        else
        {
            cout<<"No"<<endl;
        }
    }
}

素數測試