zoukankan      html  css  js  c++  java
  • POJ 2429 GCD & LCM Inverse (Pollard rho整数分解+dfs枚举)

    题意:给出a和b的gcd和lcm,让你求a和b。按升序输出a和b。若有多组满足条件的a和b,那么输出a+b最小的。
    思路:lcm=a*b/gcd
         lcm/gcd=a/gcd*b/gcd
    可知a/gcd与b/gcd互质,由此我们可以先用Pollard_rho法对lcm/gcd进行整数分解,
    然后对其因子进行深搜找出符合条件的两个互质的因数,然后再都乘以gcd即为输出答案。

    #include <iostream>
    #include <stdio.h>
    #include <algorithm>
    #include <string.h>
    #include <cstdlib>
    #include <cmath>
    
    using namespace std;
    const int maxn=100;
    long long lcm,gcd,n;
    long long factor[maxn],fac[maxn];
    long long Max;
    long long aa,bb;
    int cnt=0,len=0;
    long long multi(long long a,long long b,long long mod){
        long long ret=0;
        while(b){
            if(b&1){
                ret=(ret+a)%mod;
            }
            a=a<<1;
            a=a%mod;
            b=b>>1;
        }
        return ret;
    }
    long long quickPow(long long a,long long b,long long mod){
        long long ret=1;
        while(b){
            if(b&1)
                ret=multi(ret,a,mod);
            a=multi(a,a,mod);
            b=b>>1;
        }
        return ret;
    }
    bool witness(long long a,long long n){
        long long m=n-1;
        int j=0;
        while(!(m&1)){
            j++;
            m=m>>1;
        }
        long long x=quickPow(a,m,n);
        if(x==1||x==n-1)
            return false;
        while(j--){
            x=multi(x,x,n);
            if(x==n-1)
                return false;
        }
        return true;
    }
    //判定n是否为素数,若是,返回true
    bool Miller_Rabin(long long n){
        if(n==2)
            return true;
        if(n<2 || !(n&1))
            return false;
        for(int i=0;i<5;i++){
            long long a=rand()%(n-1)+1;
            if(witness(a,n))
                return false;
        }
        return true;
    }
    long long Gcd(long long a,long long b){
        return b==0?a:Gcd(b,a%b);
    }
    
    long long pollard_rho(long long n,int c){
        long long x,y,d,i=1,k=2;
        x=rand()%(n-1)+1;
        y=x;
        while(1){
            i++;
            x=(multi(x,x,n)+c)%n;
            d=Gcd((y-x+n)%n,n);
            if(1<d && d<n)
                return d;
            if(y==x)
                return n;
            if(i==k){
                y=x;
                k=k<<1;
            }
        }
    }
    //对n进行质因数分解
    void factorFind(long long n,int k){
        if(n==1)
            return;
        if(Miller_Rabin(n)){
            factor[cnt++]=n;
            return;
        }
        long long p=n;
        while(p>=n)
            p=pollard_rho(p,k--);
        factorFind(p,k);
        factorFind(n/p,k);
    }
    //dfs枚举a和b的值
    void dfs(long long a,long long b,int k){
        //如果目前a+b的值已经大于Max了,那么就直接return
        if(a+b>=Max){
            return;
        }
        if(k==len+1){
            if(a+b<Max){
                Max=a+b;
                aa=a;
                bb=b;
            }
            return;
        }
        dfs(a*fac[k],b,k+1);
        dfs(a,b*fac[k],k+1);
    }
    
    int main()
    {
        long long a,b;
        while(scanf("%I64d%I64d",&gcd,&lcm)!=EOF){
            n=lcm/gcd;
            //若n为素数的话,那aa=1,bb=lcm/gcd。
            if(Miller_Rabin(n)){
                //一开始WA的原因是,当gcd=1的时候,我是输出1,lcm
                //后来才发现,比如1 40,输出5 8
                aa=1;
                bb=n;
                printf("%I64d %I64d
    ",aa*gcd,bb*gcd);
            }
            else if(lcm==gcd){
                printf("%I64d %I64d
    ",gcd,gcd);
            }
            else{
                cnt=0;
                factorFind(n,120);
                sort(factor,factor+cnt);
                len=0;
                fac[0]=factor[0];
                //这里由于对n分解质因数时,a/gcd 和 b/gcd肯定两两互质,所以先将相同的素因子先乘起来
                //这样后面dfs时aa和bb肯定是两两互质的
                for(int i=1;i<cnt;i++){
                    if(factor[i]==factor[i-1]){
                        fac[len]=fac[len]*factor[i];
                    }
                    else{
                        len++;
                        fac[len]=factor[i];
                    }
                }
    
                //一开始RE,原因如下:
                //当lcm=gcd时,n=lcm/gcd=1,那么a=fac[0]=0,n/a的时候会导致RE。。。
                a=fac[0];
                b=1;
                Max=n/a+a;
                aa=a;
                bb=n/a;
                dfs(a,b,1);
                if(aa>bb){
                    long long tmp=aa;
                    aa=bb;
                    bb=tmp;
                }
                printf("%I64d %I64d
    ",aa*gcd,bb*gcd);
            }
        }
        return 0;
    }
    View Code
  • 相关阅读:
    WPF 文本滚动效果 渐变效果
    Unity3D 学习——入门资料整理
    命名管道 问题:信号灯超时问题
    Nginx 遇到的问题
    Nginx的安装配置 例子
    03 Spring的父子容器
    02 浅析Spring的AOP(面向切面编程)
    03 JVM的垃圾回收机制
    02 Java类的加载机制
    01 深入理解JVM的内存区域
  • 原文地址:https://www.cnblogs.com/chenxiwenruo/p/3567526.html
Copyright © 2011-2022 走看看