1

谁能帮我完成 pollard rho 的实施?我已经在 C 中实现了这一点。它适用于最多 10 位的数字,但它无法处理更大的数字。

请帮我改进它以执行高达 18 位的数字分解。我的代码是这样的:

#include<stdio.h>
#include<math.h>
int gcd(int a, int b)
{
    if(b==0) return a ;
    else
    return(gcd(b,a%b)) ;
}

long long int mod(long long int a , long long int b , long long int n )
{    
     long long int x=1 , y=a ;
     while(b>0)
     {
        if(b%2==1)  x = ((x%n)*(y%n))%n ;
        y = ((y%n)*(y%n))%n ;
         b/=2 ;
     }
     return x%n ;
}

int isprimes(long long int u)
{  
    if(u==3)
    return 1 ;
     int a = 2 , i ;
     long long int k , t = 0 , r , p ;
     k = u-1 ;
     while(k%2==0)
     { k/=2 ; t++ ; }

         while(a<=3)                                                              /*der are no strong pseudoprimes common in base 2 and base 3*/
         {   
         r = mod(a,k,u) ;
             for(i = 1 ; i<=t ; i++)
             {
                  p = ((r%u)*(r%u))%u ;
                  if((p==1)&&(r!=1)&&(r!=(u-1)))
                  {  return 0 ; }
                  r = p ;
             }
          if(p!=1)
          return 0 ;
         else
          a++ ;
         } 

          if(a==4)
          return 1 ;

}

long long int pol(long long int u)
{ 
  long long int x = 2 , k , i , a , y , c , s;
  int d = 1 ;
   k = 2 ;
   i = 1 ;
   y = x ;
   a = u ;
   if(isprimes(u)==1)
   { 
     return 1;
   }
   c=-1 ;
   s = 2 ;
   while(1)
   {
     i++;
     x=((x%u)*(x%u)-1)% u ;

     d = gcd(abs(y-x),u) ;

     if(d!=1&&d!=u)
     { printf("%d ",d);
       while(a%d==0) { a=a/d;  }

        x = 2 ;
        k = 2 ;
        i = 1 ;
        y = x ;
        if(a==1)
        { return 0 ; }
        if(isprimes(a)!=0)
        { return a ; }
        u=a ;

     }
     if(i==k)
     {y = x ; k*=2 ; c = x ;}                                                       /*floyd cycle detection*/
        if(c==x)                                                                 
     { x = ++s ; }
    }
    return ;

}

int main()
{
   long long int t ;
    long long int i , n , j , k , a , b  , u ;
    while(scanf("%lld",&n)&&n!=0)
    { u = n ; k = 0 ;
    while(u%2==0)
       {  u/=2 ; k = 1 ; }
      if(k==1) printf("2 ") ;
      if(u!=1)
      t = pol(u) ;
        if(u!=1) 
      {
           if(t==1)
           { printf("%lld",u) ; }
           else
           if(t!=0)
           { printf("%lld",t) ; }
      }
          printf("\n");
    }
    return 0;
}

对不起,长代码.....我是一个新的编码员。

4

1 回答 1

4

当您将两个数字相乘时m,中间产品可能会变成近似m^2。所以如果使用 64 位无符号整数类型,它可以处理的最大模数是2^32,如果模数较大,可能会发生溢出。模数稍微大一点的情况很少见,但这只会使其不那么明显,如果模数允许溢出,你不能指望幸运。

如果您选择一个代表m绝对值的残差类模数m/2或等效值,则可以将更大的范围扩大两倍:

uint64_t mod_mul(uint64_t x, uint64_t y, uint64_t m)
{
    int neg = 0;
    // if x is too large, choose m-x and note that we need one negation for that at the end
    if (x > m/2) {
        x = m - x;
        neg = !neg;
    }
    // if y is too large, choose m-y and note that we need one negation for that at the end
    if (y > m/2) {
        y = m - y;
        neg = !neg;
    }
    uint64_t prod = (x * y) % m;
    // if we had negated _one_ factor, and the product isn't 0 (mod m), negate
    if (neg && prod) {
        prod = m - prod;
    }
    return prod;
}

因此,这将允许高达2^3364 位无符号类型的模数。不是很大的一步。

该问题的推荐解决方案是使用大整数库,例如 GMP 在大多数(如果不是全部)Linux 发行版上都可以作为分发包使用,并且(相对)可以轻松安装在 Windows 上。

如果这不是一个选项(真的,你确定吗?),您可以2^63使用俄罗斯农民乘法使其适用于更大的模数(最多适用于无符号 64 位整数类型):

x * y = 2 * (x * (y/2)) + (x * (y % 2))

所以对于计算,你只需2*(m-1)要不溢出。

uint64_t mod_mult(uint64_t x, uint64_t y, uint64_t m)
{
    if (y == 0) return 0;
    if (y == 1) return x % m;
    uint64_t temp = mod_mult(x,y/2,m);
    temp = (2*temp) % m;
    if (y % 2 == 1) {
        temp = (temp + x) % m;
    }
    return temp;
}

但是请注意,此算法需要 O(log y) 步,因此在实践中相当慢。对于较小的m您可以加快速度,如果2^k*(m-1)没有溢出,您可以按k位而不是单个位 ( x*y = ((x * (y >> k)) << k) + (x * (y & ((1 << k)-1)))) 进行,如果您的模数从不大于 48 或 56 位,这是一个很好的改进。

使用模乘的变体,您的算法将适用于更大的数字(但它会明显变慢)。您还可以尝试测试模数的大小和/或因素以确定使用哪种方法,如果m < 2^32x < (2^64-1)/y,则简单即可(x * y) % m

于 2012-06-02T15:08:55.303 回答