`
刘彦明
  • 浏览: 7779 次
  • 性别: Icon_minigender_1
  • 来自: 广州
最近访客 更多访客>>
社区版块
存档分类
最新评论

大整数幂求模问题

阅读更多
      一、 问题描述:
              计算 (a^power) % m , 其中power 是非负的大整数, a, m 为大于1 的整数。

      二、 问题分析: 
        很显然, 由于 power 是大整数,因此,必须考虑到幂计算的溢出问题。怎么避免溢出呢? 可以通过降低幂次、逐次取模来实现。一个自然的想法是,将power 分成两个整数之和, power = n1 + n2,则 a^power = (a^n1) * (a^n2) .    通常采用二分法, n1 = n2 或 |n1-n2| = 1 。这就涉及到 (a * b) % m 的计算 。
   不难证明:   (a * b) % m = ((a % m) * (b % m)) % m。 
   证明如下:   设 a = pm + r1, b = qm + r2 , 则 r1 = a % m, r2 = b % m , 
                          则  (a * b) % m = (r1 * r2) % m = ((a % m) * (b % m)) % m.   证毕。

   特别地,当 a = b 时, (a^2) % m = ((a % m)^2) % m ;   这是一个简单却又关键性的结论。

三、 算法设计:

        算法1: 分治策略:
(1)若power 为奇数: 令 power = 2k+1, k 为非负整数  , 则
                   a^(2k+1) = (a^k)^2 *a ; a^(2k+1) % m = ((a^k % m)^2 % m * a) % m
  (2)   若power 为偶数:    令 power = 2k, k为非负整数, 则
                   a^(2k) = (a^k)^2 ; a^(2k) % m = (a^k % m)^2 % m
  (3)   若power == 1 : 返回 a % m ; 若 power == 0: 返回 1 % m。

       据以上(1)/(2)/(3) 条, 即可设计出相应的递归求解程序。时间复杂度为T(n) = T(n/2) + C = O(logn) ,空间复杂度为 O(1), 不足之处在于有一定的递归调用开销。

        算法2: 整数的二进制分解
将大整数 power 按照二进制进行分解:
            power = a[N] * 2^N + a[N-1] * 2^(N-1) + … + a[1] * 2 + a[0]
其中: a[i] 取值为 0 或 1 ( i=0,1,.., N),则
           a^power = a^(a[N] * 2^N) * a^(a[N-1] * 2^(N-1)) * … * a^(a[1] * 2) * a^a[0]
很显然: 
          (1) 若 a[i] = 0, 则 a[i] * 2^i = 0 , a^(a[i]*2^i) = 1, 该乘积项可以消去;
          (2) 若 a[i] = 1, 则 a[i] * 2^i = 2^i , a^(2^i) % m = (a^(2^(i-1)) % m)^2 % m. 
                令 temp = a^(2^(i-1)) % m , 则 a^(2^i) % m = (temp * temp) % m。
比如,  power = 26 = 16 + 8 + 2 = (11010)2, 则 a^26 = a^(2^4 + 2^3 + 2^1); 
计算 a^power 实际上是计算 power 的二进制表示中所有位为1对应的乘积项。
                (a^26) % m = ((a^16 %m) * (a^8 %m) * (a^2 % m)) %m
而, a^8 % m = ((a^4 %m) × (a^4%m)) % m  是可以用动态规划法逐次求解的。 简单起见, 将 (a^i) % m 称为 “取模乘积项”。

算法描述:
令 temp = a^(2^i) % m , i 是 power 的当前二进制位所对应的位置,
                                        temp 表示 power 的当前二进制位所对应的(取模)乘积项
STEP1:   初始化 temp 为 a % m  ,  result = 1;
STEP2:   对 power 进 行二进制分解。 若 power >=1 , 则进行模2运算:否则转至 STEP3
[1]  若余数为1, 则该位置上的二进制为1, 乘积中需要加入此时的 temp 项 :  result = (result * temp) % m;
           下一个二进制位对应的乘积项为 temp = (temp * temp) % m
[2]  若余数为0, 则该位置上的二进制为0,乘积中不需要加入 temp 项, result 保持不变,
           下一个二进制位对应的乘积项为 temp = (temp * temp) % m
STEP3: 所有的二进制位对应的乘积项都已计算,算法结束。

比如, result = (3^26) % 5 的计算过程如下:26 = (11010)2 ;
(1)初始化:  temp = 3^1 % 5 = 3;, result = 1 ; 
  (2)   检测 26 的最低位二进制为0,  则 不加入乘积项,result = 1,   temp =(3^2) % 5 = (temp * temp) % 5 = 4
  (3)   检测 26 的次低位二进制为1, 则 加入乘积项, result = (result * temp) % 3 = 4 , temp = (3^4) % 5 = (4*4) % 5 = 1;
  (4)   检测 26 的次低位二进制为0, 则 不加入乘积项, result = 4, temp = (3^8) % 5 = (1*1) % 5 = 1;
  (5)   检测 26 的次低位二进制为1, 则 加入乘积项, result = (result * temp) % 5 = 4, temp = (3^16) % 5 = 1;
  (6)   检测 26 的次低位二进制为1, 则 加入乘积项, result = (result * temp) % 5 = 4, temp = (3^32) % 5 = 1.
   由于所有位都检测完毕, 故 3^26 % 5 = 4.  由上可知,
   3^26 % 5 = ((3^16) % 5)) * ((3^8) % 5) * ((3^2) % 5) % 5.  其中 3^16 % 5, 3^8 % 5, 3^2 % 5 是通过动态规划法逐渐求得的。

完整的程序如下: 
程序在 Ubuntu10.10 gcc4.4.5 环境下编译运行通过。
  $ gcc -g -Wall bintmode.c runtime.c -o bintmode # 编译连接
  $ bintmode or gdb bintmode # 运行

[cpp] view plaincopyprint?
/*
* bintmode.c :  此程序计算 (a^power) % mod. power 是大整数
* 基本公式: (a*b) mod m = ((a mod m) * (b mod m)) mod m
*/ 
 
#include <stdio.h> 
#include <assert.h> 
 
int modMRec(int, int, int); 
int modMDyn(int, int ,int); 
void testModMRec(int); 
void testModMDyn(int); 
 
void testValid(int (*fun)(int a, int power, int mode)); 
void testInvalid(int (*fun)(int a, int power, int mode)); 
 
int main() 

   printf(" ******** 使用递归技术求解: ******** \n"); 
   testValid(modMRec); 
   /* testInvalid(modMRec);  */  
   defRuntime(testModMRec); 
 
   printf(" ******** 使用二进制分解技术求解: ******** \n"); 
   testValid(modMDyn); 
   /* testInvalid(modMDyn);  */  
   defRuntime(testModMDyn); 
 
   return 0; 

 
/*
* modMRec: 递归求解 (a^power) % mod , power 是个大整数
*/ 
int modMRec(int a, int power, int mod) 

   assert(a>=1 && power >=0 && mod >=1); 
   if (power == 0) { 
       return 1 % mod; 
   } 
   if (power == 1) { 
       return a % mod; 
   } 
   if (power % 2 != 0) { 
       int temp = modMRec(a, power/2, mod); 
       return  (temp * temp * a) % mod; 
   }  
   else { 
       int temp = modMRec(a, power/2, mod); 
       return  (temp * temp) % mod; 
   } 

 
/*
* modMDyn: 使用二进制分解技术求解 (a^power) % mod , power 是个大整数
*/ 
int modMDyn(int a, int power, int mod) 

   assert(a>=1 && power >=0 && mod >=1); 
   int result = 1; 
   int temp = a % mod; 
   while (power >= 1) { 
      if (power % 2 != 0) { 
         result = (result * temp) % mod;    
      } 
      temp = (temp * temp) % mod; 
      power /= 2;    
   } 
   return result; 

 
void testValid(int (*fun)(int a, int power, int mode)) 

   int base[5] = {2,3,5,7,11}; 
   int i,k; 
 
   for (i=0; i < 5; i++) { 
     for (k = 0; k < 20; k++) { 
        printf("%d ^ %d mod %d = %d .\n", base[i], k, 10, (*fun)(base[i], k, 10)); 
     }     
   } 

 
void testInvalid(int (*fun)(int a, int power, int mode)) 
{  
   printf("0 ^ 3 mod 5"); 
   (*fun)(0, 3, 5);   
   printf("3 ^ -1 mod 5"); 
   (*fun)(3, -1, 5); 
   printf("3 ^ 5 mod 0"); 
   (*fun)(3, 5, 0); 

 
void testModMRec(int n) 

   modMRec(2, n, 10); 
}   
 
void testModMDyn(int n) 

   modMDyn(2, n , 10); 


[cpp] view plaincopyprint?
#include <stdio.h> 
#include <time.h> 
 
#define MAX_SCALE 10 
 
long defScale[MAX_SCALE] = {1, 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000, 1000000000}; 
 
/* runtime: 测量 fun 指向的函数的运行时间
* fun: 指向测试函数的指针 ;
* scale:  问题规模数组 
*/ 
void runtime(void (*fun)(long scale), long* scale, int len); 
 
void defRuntime(void (*fun)(long scale)); 
 
void runtime(void (*fun)(long scale), long *scale, int len) 

   int i; 
   clock_t start, end; 
   for (i = 0; i < len; i++) { 
      start = clock(); 
      (*fun)(scale[i]); 
      end = clock(); 
      printf("scale: %12ld\t run time: %8.4f (ms). \n", scale[i], ((double)(end-start)) * 1000 / CLOCKS_PER_SEC); 
   } 

 
void defRuntime(void (*fun)(long scale)) 

   runtime(fun, defScale, MAX_SCALE); 



四、总结: 关于科学计算的算法
1. 往往要先查阅相关资料,了解相应的数学性质及结论,并对问题进行化简;例如本例中使用 (a*b)%m = ((a%m)*(b%m))%m 的模性质,避免了乘法溢出的问题。
2. 可以首选分治法和二进制分解法。分治法将所要求解的数值规模减半,而二进制分解则从数值的一个特别角度来寻求问题的解答。
分享到:
评论
2 楼 刘彦明 2013-03-29  
其实这种做法我还不太熟悉、你一下就看懂了。能给我讲下不?
1 楼 三断笛 2013-03-28  
好东西啊,检测素数会要幂模,有了博主的这些就爽了

相关推荐

    任意大整数的任意次幂

    在实际应用中,这样的大整数幂运算可以用于计算大整数的模幂,即a^n mod m,这在RSA加密算法等密码学应用中至关重要。为了进行模幂运算,程序可能还包含了取模操作,以确保结果保持在模m的范围内。 总的来说,"任意...

    浮点大数求幂(支持整数大数求幂).

    很好的体现大数模板在各种应用中的不可思议的作用。

    快速幂取模,大数幂次求模,a^p%m

    对于大整数的幂次运算与取模操作,传统的逐个相乘的方法效率低下,不适用于处理大规模数据的情况。因此,快速幂取模算法应运而生,它能够在较短的时间内完成大数的幂次计算并进行取模操作。 #### 二、算法原理 快速...

    简单幂模源代码算法(容易看懂)

    幂模运算是一种在计算大整数幂时,结合取模操作来减少计算复杂度的算法。在密码学中,特别是RSA公钥加密算法中,幂模运算起着至关重要的作用。由于直接计算大整数的幂然后取模可能会非常耗时,因此需要高效的算法来...

    易语言大数幂模运算

    在易语言中,大数幂模运算是一项重要的数学计算功能,主要涉及到大整数的乘方以及模运算。这两种运算在密码学、数学计算、编码等领域有着广泛应用。 大数幂模运算通常包括两个关键部分:大数乘方(也称幂运算)和模...

    大幂模快速算法

    求三个极大整数a,b,n的大幂模(a^b)%n。

    对整型大数的幂次求模

    快速幂算法提供了一种高效解决幂次求模问题的方法,而实现这类算法时,通常会涉及大数的加、减、乘、除等基本运算,这些运算在给定的文件代码中得到了体现。掌握这些技术和算法对于处理密码学中的大数运算至关重要,...

    128位大整数运算源代码

    大整数的乘幂运算则可能用到快速幂算法,它通过不断平方和乘法操作,大大减少了运算次数。 此外,二进制和十进制转换也是大整数处理的一部分。从10进制转2进制,可以使用除2取余法;反之,从2进制转10进制,则通常...

    模逆和模幂计算与应用

    **模幂运算**(计算 ak mod n)是指数运算在模 n 下的结果。在RSA中,公钥加密是基于幂运算,即明文 m 被加密为 c = m^e mod n,而解密使用私钥 d,c^d ≡ m mod n。在ElGamal加密中,加密同样涉及模幂运算,但解密...

    大整数运算.doc

    大整数的表示和存储可以采用数组或链表的方式,而大整数的运算实现可以采用模块化的思想,建立一个基础框架,并在此框架上讨论并实现多精度大整数的基本加法、减法、乘法、除法、平方算法、缩减、模乘、模幂乘等算法...

    一种大数模幂的快速实现方法

    这种运算涉及对非常大的整数进行幂运算后再对一个大素数取模。随着安全需求的提升,处理的数据量越来越大,传统的模幂算法在效率上往往无法满足实际应用的要求。因此,优化大数模幂运算的算法,以提高其效率,对于...

    论文研究-大整数模幂的固定基窗口组合算法.pdf

    通过采用预处理技术, 将椭圆曲线的定点标量乘的固定基窗口方法应用在模幂运算中, 与SMM算法进行组合得到一种新的求模幂乘算法——固定基窗口方法。对算法的原理与效率进行了分析, 实验结果表明, 算法的运算速度得到...

    大整数对小整数取模

    大整数对小整数取模

    大整数包的设计与运算

    在信息安全中,大整数包常用于加密算法,如RSA的模幂运算(`a^b mod m`)、模乘(`a * b mod m`)等。这些运算要求在模数m的范围内进行,且m通常是一个非常大的质数。例如,RSA的公钥和私钥就是通过大整数的因子分解...

    RSA算法的纯Python实现

    这个库是计算乘模运算,幂模运算(蒙哥马利算法),最大公约数算法及扩展最大公约数算法(扩展欧几里得算法)等。 2、质数库。Miller_Rabin素数判断法,大整数快速因式分解算法(pollard_rho算法),生成指定位数的...

    模幂算法的很多实现方式

    Montgomery模幂算法是一种在大整数运算中常用于优化模幂运算的方法,特别适合于硬件实现。它主要通过对模运算进行预处理,将模除操作转化为乘法,从而避免了昂贵的除法操作。Montgomery算法在RSA加密等密码学应用中...

    密码学 模逆与模幂计算与应用 实验五报告

    模幂运算在加密过程中用于快速提升密钥的幂次,通常采用分治策略的右向左或左向右幂算法,可以高效地计算大整数的幂。实验要求计算不超过216的三个正整数a、e和n的模幂,这在RSA的加密和解密过程中至关重要。 3. **...

    C#的大整数实现

    在编程领域,大整数(BigInteger)是指超出标准整型数据类型表示范围的整数。在C#中,虽然内置的`int`、`long`等类型可以处理大部分日常...无论哪种方式,理解和熟练使用大整数对于解决涉及大数值的复杂问题至关重要。

Global site tag (gtag.js) - Google Analytics