1.Miller-rabin算法:

Miller-rabin算法是一个用来快速判断一个正整数是否为素数的算法。

根据费马小定理,如果p是素数,则a^(p-1)≡1(mod p)对所有的a∈[1,n-1]成立。所以如果在[1,n-1]中随机取出一个a,发现不满足费马小定理,则证明n必为合数。

【但是每次尝试过程中还做了一个优化操作,以提高用少量的a检测出p不是素数的概率。这个优化叫做二次探测。它是根据这个定理:如果p是一个素数,那么对于x(0<x<p),若x^2%p=1,则x=1或p-1。】

为了计算a^(n-1)mod n,我们把n-1分解为x* 2^t的形式,其中t>=1且x是奇数;因此,a^(n-1)≡(a^x)^(2^t)(mod n),所以可以通过先计算a^x mod n,然后对结果连续平方t次来计算a^(n-1) mod n。一旦发现某次平方后mod n等于1了,那么说明符合了二次探测定理的逆否命题使用条件,立即检查x是否等于1或n-1,如果不等于1也不等于n-1则可直接判定p为合数。

2.pollard-rho算法:

这是一个用来快速对整数进行质因数分解的算法,需要与Miller-rabin共同使用。

算法原理:

1.通过某种方法得到两个整数a和b,而待分解的大整数为n。

2.计算p=gcd(a-b,n),直到p不为1(就是a-b与n不是互质),或者a,b出现循环为止。

3.然后再判断p=n?

4.如果p=n,那么返回n是一个质数。

5.否则返回p是n的一个因子,那么我们又可以递归的计算Pollard(p)和Pollard(n/p),这样,我们就可以求出n的所有质因子。

算法步骤:选取一个小的随机数x1,迭代生成x[i] = x[i-1]^2+c,一般取c=1,若序列出现循环则退出,计算p=gcd(x[i-1]-x[i],n),若p=1则返回上一步继续迭代,否则跳出迭代过程。若p=n,则n为素数,否则p为n的一个约数,并递归分解p和n/p。

【小知识】:随机数生成

C++中函数srand(),可以指定不同的数(无符号整数变元)为种子。但是如果种子相同,伪随机数列也相同。

比较理想的是用变化的数,比如时间来作为随机数生成器的种子。 time的值每时每刻都不同,即种子不同,所以,产生的随机数也不同。

用法什么的想深入了解自己去搜吧,这里只要明白下面的程序中随机数是这样产生的就行了。然后,在这里再举个小栗子以加深一下对它的理解:

  1. #include <cstdio>
  2. #include <iostream>
  3. #include <cstring>
  4. #include <algorithm>
  5. #include <cmath>
  6. #include <ctime>//这个必须有
  7. using namespace std;
  8. int main()
  9. {
  10. int a = 100;
  11. srand( time( NULL));
  12. while(a--)
  13. cout << rand() << endl;
  14. return 0;
  15. }
  16. //这个程序的作用是产生100个随机数
  17. //如果你和我一样有颗童心去多试几次的话你会发现——每次产生的随机数都不一样
  18. //噫 是不是狠有趣(。^▽^)

学了这么多是不是手痒了?别着急,点我有惊喜。

AC代码(C++【因为涉及到ctime,所以G++会RE的】):

  1. /* *************************************************
  2. *
  3. * Miller_Rabin 算法进行素数测试 
  4. * 速度快,可以判断一个 < 2^63 的数是不是素数
  5. *
  6. **************************************************/
  7. #include <cstdio>
  8. #include <iostream>
  9. #include <cstring>
  10. #include <algorithm>
  11. #include <cmath>
  12. #include <ctime>
  13. using namespace std;
  14. const int S = 8; //随机算法判定次数,一般8~10次就够了
  15. //计算ret = (a*b)%c a, b, c < 2^63
  16. long long mult_mod(long long a, long long b, long long c)
  17. {
  18. a %= c;
  19. b %= c;
  20. long long ret = 0;
  21. long long tem = a;
  22. while(b)
  23. {
  24. if(b & 1)
  25. {
  26. ret += tem;
  27. if(ret > c) ret -= c; //直接取模慢很多
  28. }
  29. tem <<= 1;
  30. if(tem > c) tem -= c;
  31. b >>= 1;
  32. }
  33. return ret;
  34. }
  35. //计算 ret = (a^n) % mod
  36. long long pow_mod(long long a, long long n, long long mod)
  37. {
  38. long long ret = 1;
  39. long long tem = a % mod;
  40. while(n)
  41. {
  42. if(n & 1) ret = mult_mod(ret, tem, mod);
  43. tem = mult_mod(tem, tem, mod);
  44. n >>= 1;
  45. }
  46. return ret;
  47. }
  48. // 通过 a^(n-1)=1(mod n)来判断n是不是素数
  49. // n-1 = x * 2^t 中间使用二次判断
  50. // 是合数返回true,不一定是合数返回false
  51. bool check(long long a, long long n, long long x, long long t)
  52. {
  53. long long ret = pow_mod(a, x, n); //a^x % n
  54. long long last = ret;
  55. for( int i = 1; i <= t; ++i) //进行t次(a^x % n)^2 % n
  56. {
  57. ret = mult_mod(ret, ret, n);
  58. if(ret == 1 && last != 1 && last != n - 1) return true; //合数
  59. last = ret;
  60. }
  61. if(ret != 1) return true;
  62. return false; //不一定是合数
  63. }
  64. //**************************************************
  65. // Miller_Rabin算法
  66. // 是素数返回true,(可能是伪素数)
  67. // 不是素数返回false
  68. //**************************************************
  69. bool Miller_Rabin(long long n)
  70. {
  71. if(n < 2) return false;
  72. if(n == 2) return true;
  73. if( (n& 1) == 0) return false; //偶数
  74. long long x = n - 1;
  75. long long t = 0;
  76. while( (x& 1) == 0) //将n分解为x*2^t;
  77. {
  78. x >>= 1;
  79. t++;
  80. }
  81. srand( time( NULL));
  82. for( int i = 0; i < S; ++i)
  83. {
  84. long long a = rand()%(n -1) + 1; //产生随机数a(并控制其范围在1 ~ n-1之间)
  85. if(check(a, n, x, t)) //是合数
  86. return false;
  87. }
  88. return true;
  89. }
  90. //**********************************************  
  91. //  
  92. // pollard_rho 算法进行质因素分解  
  93. //  
  94. //********************************************* 
  95. int tol; //质因数的个数,编号为0~tol-1
  96. long long factor[ 100]; //质因素分解结果(刚返回时是无序的)
  97. long long gcd(long long a, long long b)
  98. {
  99. long long t;
  100. while(b)
  101. {
  102. t = a;
  103. a = b;
  104. b = t % b;
  105. }
  106. if(a >= 0) return a;
  107. return -a;
  108. }
  109. //找出一个因子
  110. long long pollard_rho(long long x, long long c)
  111. {
  112. long long i = 1, k = 2;
  113. srand( time( NULL));
  114. long long x0 = rand()%(x -1) + 1; //产生随机数x0(并控制其范围在1 ~ x-1之间)
  115. long long y = x0;
  116. while( 1)
  117. {
  118. i++;
  119. x0 = (mult_mod(x0, x0, x) + c) % x;
  120. long long d = gcd(y - x0, x);
  121. if(d != 1 && d != x) return d;
  122. if(y == x0) return x;
  123. if(i == k)
  124. {
  125. y = x0;
  126. k += k;
  127. }
  128. }
  129. }
  130. //对n进行素因子分解,存入factor。 k设置为107左右即可
  131. void findfac(long long n, int k)
  132. {
  133. if(n == 1) return ;
  134. if(Miller_Rabin(n)) //是素数就把这个素因子存起来
  135. {
  136. factor[tol++] = n;
  137. return ;
  138. }
  139. int c = k;
  140. long long p = n;
  141. while(p >= n)
  142. p = pollard_rho(p, c--); //值变化,防止陷入死循环k
  143. findfac(p, k);
  144. findfac(n/p, k);
  145. }
  146. int main()
  147. {
  148. int T;
  149. long long n;
  150. scanf( "%d",&T);
  151. while(T--)
  152. {
  153. scanf( "%lld",&n);
  154. if(Miller_Rabin(n)) cout << "Prime" << endl;
  155. else
  156. {
  157. tol = 0;
  158. findfac(n, 107);
  159. long long ans = factor[ 0];
  160. for( int i = 1; i < tol; ++i)
  161. ans = min(ans, factor[i]);
  162. cout << ans << endl;
  163. }
  164. }
  165. return 0;
  166. }

经过了“几天”的学习,终于能明白M-r,p-r算法是怎么实现的了(吐槽一下那三位对我关爱有加的兄弟,给我留了个这么有用的知识点让我讲),然后对着板子敲了几遍熟悉了一下。可能是还没碰到这种题目吧,内心里总觉得。。?总之也算是没浪费这些时间,至少我可以对着板子来对这知识点进行自在应用了(比如用朴素算法一不小心就会超时的一道题目。

【知识点】大数分解与素数判定 --- 【Miller-rabin算法】【pollard-rho算法】相关推荐

  1. 大整数分解——Pollard Rho算法

    延续上一篇,这次来讲一讲大整数分解算法的应用. 要解决的问题很简单,对一个整数进行分解质因数. 首先还是效率非常低的暴力算法,相信大家都会,不多提. 和上次一样,当数达到非常大的时候,分解将变得非常困 ...

  2. 大素数分解与素数判定

    1.Miller-rabin算法: Miller-rabin算法是一个用来快速判断一个正整数是否为素数的算法. 根据费马小定理,如果p是素数,则a^(p-1)≡1(mod p)对所有的a∈[1,n-1 ...

  3. 数学--数论--随机算法--Pollard Rho 大数分解算法 (带输出版本)

    RhoPollard Rho是一个著名的大数质因数分解算法,它的实现基于一个神奇的算法:MillerRabinMillerRabin素数测试. 操作流程 首先,我们先用MillerRabinMille ...

  4. 数学--数论--随机算法--Pollard Rho 大数分解算法(纯模板带输出)

    ACM常用模板合集 #include <bits/stdc++.h> using namespace std; typedef long long ll; ll pr; ll pmod(l ...

  5. 素数判定质因数分解(数论)(Miller Rabin)(Pollard Rho)

    太玄学了! 我真的被概率的魅力折服了.此前我认为1便是1,0.9999999999-便是0.9999999999-. 但实际上它们有着千丝万缕的关系. 试想,如果一件事发生的概率是0.99999999 ...

  6. 大素数判定以及大整数分解

    素数判定--Miller Rabin 算法 #include <cstdio> #include <cstring> #include <cmath> #inclu ...

  7. 数学--数论---P4718 Pollard-Rho算法 大数分解

    P4718 [模板]Pollard-Rho算法 题目描述 MillerRabin算法是一种高效的质数判断方法.虽然是一种不确定的质数判断法,但是在选择多种底数的情况下,正确率是可以接受的.Pollar ...

  8. 简述大数分解算法Pollard Rho和Pollard p-1

    大数分解问题其实至今都是一个世界级难题,最常见的分解法是从2一直找到sqr(N),作为一个密码学专业的学生,每次看到别人这么做来进行因子分解,自己都控制不住想要制止他,因为这个算法的效率简直太太太太太 ...

  9. 素数判定算法 MILLER RABIN

    入门级筛素数--试除法,复杂度O(n^2) bool rmprime( long long n ) {for(long long i = 2; i <= sqrt(n) ; i++) if(n% ...

最新文章

  1. 谢文睿:西瓜书 + 南瓜书 吃瓜系列 1. 一元线性回归
  2. 表 合并字段_多工作表动态合并,其实很简单
  3. 如何在Exchange中处理不能发送的信息?
  4. matlab 通过矩阵变换使图像旋转平移_数字图像处理|P11 第三章 第四节 图像的几何变换...
  5. 【图示,简单明了】HttpServlet中getAttribute和getParameter的区别——【javaweb系列学习笔记】
  6. python处理excel可视化_python如何将excel数据处理可视化
  7. 使用Docker容器和Java EE进行持续交付
  8. 2021年上半年系统集成项目管理工程师综合知识真题及答案解析
  9. Catlyst 6509告警信息--把trunk口配置成access
  10. Theano 中文文档 0.9 - 5.1 Ubuntu安装说明
  11. 华为海思Hi1620芯片发布在即 7nm制程ARM架构最高可达3.0GHz
  12. 5.4.2 Using Hibernate templates
  13. c语言ch能储存多少字符,二级c语言程序设计习题与解答ch5-7函数字符(11页)-原创力文档...
  14. Dropout抑制过拟合
  15. 大型网站的系统架构(摘)
  16. HTML跳转页面指定位置的几种方法
  17. springSessionDemo
  18. 基于91助手实现80/54坐标转换到2000大地坐标的七参数计算
  19. html 滚动条设置透明,iframe 透明 改变iframe滚动条..设置透明
  20. 利用C#实现的外挂式甲骨文拼音输入法

热门文章

  1. Matlab在线IDE:MATLAB Online介绍与计算定积分案例
  2. 生成普通二维码测试例子
  3. 【Twomon SE】让你的ipad 变成电脑副屏 提高代码效率
  4. 【JAVA】贪吃蛇的初步实现(二)
  5. echarts添加参考线以及清除参考线。
  6. python创建文件、写入文件、读取文件
  7. Mybatis框架详细介绍
  8. 10大H5前端框架,让你开发不愁
  9. C++ 删除指定字符串中的某些字符
  10. Thinkphp6 使用composer 安装