51Nod 1135-原根(快速求解一个素数的原根)

来源:互联网 发布:mac os dmg镜像下载 编辑:程序博客网 时间:2024/05/29 11:47

题目地址:51Nod 1135
1.原根定义:设m>1,gcd(a,m)=1,使得这里写图片描述成立的最小的r,称为a对模m的阶。
2.定理:如果模m有原根,那么他一共有这里写图片描述个原根。
3.定理:如果p为素数,那么素数p一定存在原根,并且模p的原根的个数为这里写图片描述个。
4.定理:假设m是正整数,a是整数,如果a模m的阶等于这里写图片描述,则称a为模m的一个原根。
5.模m有原根的充要条件:m=2,4,P^a,2*P^a…….
求模素数P的原根的方法:对P-1素因子分解,即P-1=(P1^a1)(P2^a2)…..(Pk^ak)。,若恒有这里写图片描述成立,那么g就是P的原根(对于合数而言,只需要把p-1换成这里写图片描述即可)

#include <stdio.h>#include <math.h>#include <string.h>#include <stdlib.h>#include <iostream>#include <sstream>#include <algorithm>#include <set>#include <queue>#include <stack>#include <map>#include <bitset>#pragma comment(linker, "/STACK:102400000,102400000")using namespace std;typedef long long  LL;const int inf=0x3f3f3f3f;const double pi= acos(-1.0);const double esp=1e-7;const int Maxn=1e6+10;int prime[Maxn];//存储素数int sprime[Maxn];//存储P-1的素因子bitset<Maxn>pri;//结果只有0和1,判断是否为素数int k;//记录Maxn以内的素数个数int cnt;//记录素因子的个数void is_prime(){    pri.set();//将所有的二进制数都标为1    for(int i=2; i<Maxn; i++) {        if(pri[i]) {            prime[k++]=i;            for(int j=i+i; j<Maxn; j+=i)                pri[j]=0;        }    }}void Divide(int n)//将n分解为素因子{    cnt=0;    int t=(int)sqrt(1.0*n);    for(int i=0; prime[i]<=t; i++) {        if(n%prime[i]==0) {            sprime[cnt++]=prime[i];            while(n%prime[i]==0)//因为有可能有多个peime[i]                n/=prime[i];        }    }    if(n>1)        sprime[cnt++]=n;//可能只有自己一个素因子}LL modexp(LL a,LL b,int mod)//快速幂取余{    LL res=1;    while(b>0) {        a=a%mod;        if(b&1)            res=res*a%mod;        b=b>>1;        a=a*a%mod;    }    return res;}int main(){    int p;    is_prime();    while(~scanf("%d",&p)) {        Divide(p-1);        for(int g=2; g<p; g++) {            int flag=1;            for(int i=0; i<cnt; i++) {                int t=(p-1)/sprime[i];                if(modexp(g,t,p)==1) {                    flag=0;                    break;                }            }            if(flag) {                int root=g;                printf("%d\n",root);                break;//去掉break的话是求所有的原根,加上break是求最小的原根、            }        }    }    return 0;}
1 0
原创粉丝点击