zoukankan      html  css  js  c++  java
  • HDU3930(离散对数与原根)

    题目:Broot


    题意:给出k,m,newx的值,求方程x^k(mod m)=newx的解,其中m为素数。


    解法步骤:

    (1)先暴力求m的原根g

    (2)大步小步求g^t1(mod m)=newx

    (3)则g^(t1+n*t2)(mod m)=newx,t2=m-1

    (4)x=g^y(mod m),x^k=(g^y)^k=g^(yk)=g^(t1+n*t2)

    那么就是求同于方程yk=t1(mod t2),求出y之后带入x=g^y(mod m)解出x


    #include <iostream>
    #include <string.h>
    #include <algorithm>
    #include <stdio.h>
    #include <math.h>
    #include <map>
    
    using namespace std;
    typedef long long LL;
    
    const int N=1000005;
    
    int p[N];
    bool prime[N];
    int num,cnt;
    LL k,m,newx,g;
    LL a[65],b[65];
    
    void isprime()
    {
        num=0;
        int i,j;
        memset(prime,true,sizeof(prime));
        for(i=2;i<N;i++)
        {
            if(prime[i])
            {
                p[num++]=i;
                for(j=i+i;j<N;j+=i)
                {
                    prime[j]=false;
                }
            }
        }
    }
    
    LL multi(LL a,LL b,LL m)
    {
        LL ans=0;
        while(b)
        {
            if(b&1)
            {
                ans=(ans+a)%m;
                b--;
            }
            b>>=1;
            a=(a+a)%m;
        }
        return ans;
    }
    
    LL quick_mod(LL a,LL b,LL m)
    {
        LL ans=1;
        a%=m;
        while(b)
        {
            if(b&1)
            {
                ans=multi(ans,a,m);
                b--;
            }
            b>>=1;
            a=multi(a,a,m);
        }
        return ans;
    }
    
    void factor(LL n)
    {
        cnt=0;
        for(int i=0;(LL)p[i]*p[i]<=n;i++)
        {
            if(n%p[i]==0)
            {
                a[cnt]=p[i];
                int c=0;
                while(n%p[i]==0)
                {
                    c++;
                    n/=p[i];
                }
                b[cnt++]=c;
            }
        }
        if(n>1)
        {
            a[cnt]=n;
            b[cnt++]=1;
        }
    }
    
    LL extend_Euclid(LL a,LL b,LL &x,LL &y)
    {
        if(b==0)
        {
            x=1;
            y=0;
            return a;
        }
        LL gd=extend_Euclid(b,a%b,x,y);
        LL temp=x;
        x=y;
        y=temp-(a/b)*y;
        return gd;
    }
    
    LL Inv(LL n,LL p)
    {
        return quick_mod(n,p-2,p);
    }
    
    bool dfs(int dept,LL t)
    {
        if(dept==cnt)
        {
            LL ans=quick_mod(g,t,m);
            if(ans==1&&t!=m-1) return false;
            return true;
        }
        LL tmp=1;
        for(int i=0;i<=b[dept];i++)
        {
            if(!dfs(dept+1,t*tmp)) return false;
            tmp*=a[dept];
        }
        return true;
    }
    
    void find()
    {
        factor(m-1);
        for(g=2;;g++)
            if(dfs(0,1)) break;
    }
    
    LL log_x(LL a,LL b,LL p)
    {
        map<LL,int>x;
        LL z=(LL)ceil(sqrt(p*1.0));
        LL v=Inv(quick_mod(a,z,p),p);
        LL e=1;
        x[1]=0;
        for(int i=1;i<z;i++)
        {
            e=multi(e,a,p);
            if(!x.count(e))
                x[e]=i;
        }
        for(int i=0;i<z;i++)
        {
            if(x.count(b))
                return i*z+x[b];
            b=multi(b,v,p);
        }
        return -1;
    }
    
    LL sol[1005];
    
    void Solve(LL a,LL b,LL n)
    {
        LL d,x,y;
        d=extend_Euclid(a,n,x,y);
        if(b%d) puts("-1");
        else
        {
            n/=d;b/=d;
            sol[0]=(x*b%n+n)%n;
            for(int i=1;i<d;i++)
                sol[i]=sol[i-1]+n;
            for(int i=0;i<d;i++)
                sol[i]=quick_mod(g,sol[i],m);
            sort(sol,sol+d);
            for(int i=0;i<d;i++)
                cout<<sol[i]<<endl;
        }
    }
    
    int main()
    {
        int t=1;
        isprime();
        while(cin>>k>>m>>newx)
        {
            find();
            LL t1=log_x(g,newx,m);
            LL t2=m-1;
            cout<<"case"<<t++<<":"<<endl;
            Solve(k,t1,t2);
        }
        return 0;
    }
    


  • 相关阅读:
    git 获取领先落后的命令
    django orm 时间字段讲解
    在SAE上同步djanogo的mysql数据库
    使用JS来实现验证码功能
    一个基于python的即时通信程序
    关于python多线程编程中join()和setDaemon()的一点儿探究
    Django1.6添加comments应用的简单过程
    使用saltstack批量部署服务器运行环境事例——批量部署nagios客户端
    Web服务器集群搭建关键步骤纪要
    关于rsync的密码问题
  • 原文地址:https://www.cnblogs.com/jiangu66/p/3221520.html
Copyright © 2011-2022 走看看