• 51nod 1135 原根



    基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题
     收藏
     关注
    设m是正整数,a是整数,若a模m的阶等于φ(m),则称a为模m的一个原根。(其中φ(m)表示m的欧拉函数)
    给出1个质数P,找出P最小的原根。
    Input
    输入1个质数P(3 <= P <= 10^9)
    Output
    输出P最小的原根。
    Input示例
    3
    Output示例
    2

    1.原根定义:设m>1,gcd(a,m)=1,使得这里写图片描述成立的最小的r,称为a对模m的阶。 
    2.定理:如果模m有原根,那么他一共有这里写图片描述个原根。这里的函数表示[1,m)中与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
    ",root);
                    break;//去掉break的话是求所有的原根,加上break是求最小的原根、
                }
            }
        }
        return 0;
    }
    









  • 相关阅读:
    双USB墙壁电源插座面板....制作详解-电子产品世界论坛
    公共充电站?不用守着手机充电了 | 爱范儿
    公司简介-远能电气
    连接器|网络滤波连接器|电脑连接器|RJ45变压器-华联威电子有限公司
    自己动手做简单移动电源
    移动电源市场有多乱?
    聚合物电池_百度百科
    电芯:聚合物的一定比18650更好_移动电源_移动电源评测-中关村在线
    图文:充电宝租借自助机现身江城_新浪财经_新浪网
    不按时归还,信用会有“污点”_新浪浙江_新浪网
  • 原文地址:https://www.cnblogs.com/bryce1010/p/9387108.html
Copyright © 2020-2023  润新知