当前位置 : 主页 > 网络编程 > PHP >

逆元详解

来源:互联网 收集:自由互联 发布时间:2023-09-06
今天我们来探讨逆元在ACM-ICPC竞赛中的应用,逆元是一个很重要的概念,必须学会使用它。 对于正整数 和 ,如果有 ,那么把这个同余方程中 的最小正整数解叫做 模 的逆元。 逆元一般




今天我们来探讨逆元在ACM-ICPC竞赛中的应用,逆元是一个很重要的概念,必须学会使用它。


对于正整数

逆元详解_#include


逆元详解_#include_02

,如果有

逆元详解_i++_03

,那么把这个同余方程中

逆元详解_i++_04

的最小正整数解叫做

逆元详解_#include


逆元详解_#include_02

的逆元。


逆元一般用扩展欧几里得算法来求得,如果

逆元详解_#include_02

为素数,那么还可以根据费马小定理得到逆元为

逆元详解_#include_08



推导过程如下

                            

逆元详解_i++_09

 

求现在来看一个逆元最常见问题,求如下表达式的值(已知

逆元详解_i++_10


         

逆元详解_#include_11

  


当然这个经典的问题有很多方法,最常见的就是扩展欧几里得,如果

逆元详解_#include_02

是素数,还可以用费马小定理。


但是你会发现费马小定理和扩展欧几里得算法求逆元是有局限性的,它们都会要求

逆元详解_#include


逆元详解_#include_02

互素。实际上我们还有一

种通用的求逆元方法,适合所有情况。公式如下


          

逆元详解_ios_15


现在我们来证明它,已知

逆元详解_i++_10

,证明步骤如下


          

逆元详解_i++_17


接下来来实战一下,看几个关于逆元的题目。


题目:​​http://poj.org/problem?id=1845​​

​ 

题意:给定两个正整数

逆元详解_ios_18


逆元详解_#include_19

,求

逆元详解_i++_20

的所有因子和对9901取余后的值。


分析:很容易知道,先把

逆元详解_ios_18

分解得到

逆元详解_i++_22

,那么得到

逆元详解_ios_23

,那么

逆元详解_i++_20

     的所有因子和的表达式如下


    

逆元详解_i++_25


所以我们有两种做法。第一种做法是二分求等比数列之和。

 

代码:


[cpp]  ​​view plain​​  ​​copy​​

  ​​​ ​​​

  1. #include <iostream>  
  2. #include <string.h>  
  3. #include <stdio.h>  
  4.   
  5. using namespace std;  
  6. typedef long long LL;  
  7. const int N = 10005;  
  8. const int MOD = 9901;  
  9.   
  10. bool prime[N];  
  11. int p[N];  
  12. int cnt;  
  13.   
  14. void isprime()  
  15. {  
  16.     cnt = 0;  
  17. true,sizeof(prime));  
  18. for(int i=2; i<N; i++)  
  19.     {  
  20. if(prime[i])  
  21.         {  
  22.             p[cnt++] = i;  
  23. for(int j=i+i; j<N; j+=i)  
  24. false;  
  25.         }  
  26.     }  
  27. }  
  28.   
  29. LL power(LL a,LL b)  
  30. {  
  31.     LL ans = 1;  
  32.     a %= MOD;  
  33. while(b)  
  34.     {  
  35. if(b & 1)  
  36.         {  
  37.             ans = ans * a % MOD;  
  38.             b--;  
  39.         }  
  40.         b >>= 1;  
  41.         a = a * a % MOD;  
  42.     }  
  43. return ans;  
  44. }  
  45.   
  46. LL sum(LL a,LL n)  
  47. {  
  48. if(n == 0) return 1;  
  49.     LL t = sum(a,(n-1)/2);  
  50. if(n & 1)  
  51.     {  
  52.         LL cur = power(a,(n+1)/2);  
  53.         t = (t + t % MOD * cur % MOD) % MOD;  
  54.     }  
  55. else  
  56.     {  
  57.         LL cur = power(a,(n+1)/2);  
  58.         t = (t + t % MOD * cur % MOD) % MOD;  
  59.         t = (t + power(a,n)) % MOD;  
  60.     }  
  61. return t;  
  62. }  
  63.   
  64. void Solve(LL A,LL B)  
  65. {  
  66.     LL ans = 1;  
  67. for(int i=0; p[i]*p[i] <= A; i++)  
  68.     {  
  69. if(A % p[i] == 0)  
  70.         {  
  71. int num = 0;  
  72. while(A % p[i] == 0)  
  73.             {  
  74.                 num++;  
  75.                 A /= p[i];  
  76.             }  
  77.             ans *= sum(p[i],num*B) % MOD;  
  78.             ans %= MOD;  
  79.         }  
  80.     }  
  81. if(A > 1)  
  82.     {  
  83.         ans *= sum(A,B) % MOD;  
  84.         ans %= MOD;  
  85.     }  
  86.     cout<<ans<<endl;  
  87. }  
  88.   
  89. int main()  
  90. {  
  91.     LL A,B;  
  92.     isprime();  
  93. while(cin>>A>>B)  
  94.         Solve(A,B);  
  95. return 0;  
  96. }  



第二种方法就是用等比数列求和公式,但是要用逆元。用如下公式即可


                     

逆元详解_ios_15

 

因为

逆元详解_#include_27

可能会很大,超过int范围,所以在快速幂时要二分乘法。


代码:


[cpp]  ​​view plain​​  ​​copy​​

  ​​​ ​​​

  1. #include <iostream>  
  2. #include <string.h>  
  3. #include <stdio.h>  
  4.   
  5. using namespace std;  
  6. typedef long long LL;  
  7. const int N = 10005;  
  8. const int MOD = 9901;  
  9.   
  10. bool prime[N];  
  11. int p[N];  
  12. int cnt;  
  13.   
  14. void isprime()  
  15. {  
  16.     cnt = 0;  
  17. true,sizeof(prime));  
  18. for(int i=2; i<N; i++)  
  19.     {  
  20. if(prime[i])  
  21.         {  
  22.             p[cnt++] = i;  
  23. for(int j=i+i; j<N; j+=i)  
  24. false;  
  25.         }  
  26.     }  
  27. }  
  28.   
  29. LL multi(LL a,LL b,LL m)  
  30. {  
  31.     LL ans = 0;  
  32.     a %= m;  
  33. while(b)  
  34.     {  
  35. if(b & 1)  
  36.         {  
  37.             ans = (ans + a) % m;  
  38.             b--;  
  39.         }  
  40.         b >>= 1;  
  41.         a = (a + a) % m;  
  42.     }  
  43. return ans;  
  44. }  
  45.   
  46. LL quick_mod(LL a,LL b,LL m)  
  47. {  
  48.     LL ans = 1;  
  49.     a %= m;  
  50. while(b)  
  51.     {  
  52. if(b & 1)  
  53.         {  
  54.             ans = multi(ans,a,m);  
  55.             b--;  
  56.         }  
  57.         b >>= 1;  
  58.         a = multi(a,a,m);  
  59.     }  
  60. return ans;  
  61. }  
  62.   
  63. void Solve(LL A,LL B)  
  64. {  
  65.     LL ans = 1;  
  66. for(int i=0; p[i]*p[i] <= A; i++)  
  67.     {  
  68. if(A % p[i] == 0)  
  69.         {  
  70. int num = 0;  
  71. while(A % p[i] == 0)  
  72.             {  
  73.                 num++;  
  74.                 A /= p[i];  
  75.             }  
  76.             LL M = (p[i] - 1) * MOD;  
  77.             ans *= (quick_mod(p[i],num*B+1,M) + M - 1) / (p[i] - 1);  
  78.             ans %= MOD;  
  79.         }  
  80.     }  
  81. if(A > 1)  
  82.     {  
  83.         LL M = MOD * (A - 1);  
  84.         ans *= (quick_mod(A,B+1,M) + M - 1) / (A - 1);  
  85.         ans %= MOD;  
  86.     }  
  87.     cout<<ans<<endl;  
  88. }  
  89.   
  90. int main()  
  91. {  
  92.     LL A,B;  
  93.     isprime();  
  94. while(cin>>A>>B)  
  95.         Solve(A,B);  
  96. return 0;  
  97. }  



 

其实有些题需要用到

逆元详解_ios_28


逆元详解_i++_29

的所有逆元,这里

逆元详解_i++_29

为奇质数。那么如果用快速幂求时间复杂度为

逆元详解_ios_31

,如果对于一个1000000级别的素数

逆元详解_i++_29

,这样做的时间复杂度是很高了。实际上有

逆元详解_#include_33

的算法,有一个递推式如下


                   

逆元详解_#include_34

 

它的推导过程如下,设

逆元详解_ios_35

,那么


       

逆元详解_i++_36


对上式两边同时除

逆元详解_i++_37

,进一步得到


       

逆元详解_#include_38


再把

逆元详解_i++_39


逆元详解_ios_40

替换掉,最终得到


       

逆元详解_#include_34


初始化

逆元详解_i++_42

,这样就可以通过递推法求出

逆元详解_ios_28

模奇素数

逆元详解_i++_29

的所有逆元了。


另外

逆元详解_ios_28


逆元详解_i++_29

的所有逆元值对应

逆元详解_ios_28

中所有的数,比如

逆元详解_#include_48

,那么

逆元详解_ios_49

对应的逆元是

逆元详解_#include_50




题目:​​http://www.lydsy.com/JudgeOnline/problem.php?id=2186​​

​ 

题意:求

逆元详解_#include_51


逆元详解_ios_52

互质的数的个数,其中

逆元详解_ios_53



分析:因为

逆元详解_ios_53

,所以

逆元详解_#include_55

,我们很容易知道如下结论

     对于两个正整数

逆元详解_i++_56

逆元详解_#include_57

,如果

逆元详解_#include_57

逆元详解_i++_56

的倍数,那么

逆元详解_ios_60

中与

逆元详解_i++_56

互素的数的个数为

逆元详解_ios_62

​ 

     本结论是很好证明的,因为

逆元详解_ios_63

中与

逆元详解_i++_56

互素的个数为

逆元详解_i++_65

,又知道

逆元详解_ios_66

,所以

     结论成立。那么对于本题,答案就是


     

逆元详解_#include_67


      其中

逆元详解_#include_68

为小于等于

逆元详解_#include_69

的所有素数,先筛选出来即可。由于最终答案对一个质数取模,所以要用逆元,这里

      求逆元就有技巧了,用刚刚介绍的递推法预处理,否则会TLE的。


代码:


[cpp]  ​​view plain​​  ​​copy​​

  ​​​ ​​​

  1. #include <iostream>  
  2. #include <string.h>  
  3. #include <stdio.h>  
  4. #include <bitset>  
  5.   
  6. using namespace std;  
  7. typedef long long LL;  
  8. const int N = 10000005;  
  9.   
  10. bitset<N> prime;  
  11.   
  12. void isprime()  
  13. {  
  14.     prime.set();  
  15. for(int i=2; i<N; i++)  
  16.     {  
  17. if(prime[i])  
  18.         {  
  19. for(int j=i+i; j<N; j+=i)  
  20. false;  
  21.         }  
  22.     }  
  23. }  
  24.   
  25. LL ans1[N],ans2[N];  
  26. LL inv[N];  
  27.   
  28. int main()  
  29. {  
  30.     isprime();  
  31. int MOD,m,n,T;  
  32. "%d%d",&T,&MOD);  
  33.     ans1[0] = 1;  
  34. for(int i=1; i<N; i++)  
  35.         ans1[i] = ans1[i-1] * i % MOD;  
  36.     inv[1] = 1;  
  37. for(int i=2;i<N;i++)  
  38.     {  
  39. if(i >= MOD) break;  
  40.         inv[i] = (MOD - MOD / i) * inv[MOD % i] % MOD;  
  41.     }  
  42.     ans2[1] = 1;  
  43. for(int i=2; i<N; i++)  
  44.     {  
  45. if(prime[i])  
  46.         {  
  47.             ans2[i] = ans2[i-1] * (i - 1) % MOD;  
  48.             ans2[i] = ans2[i] * inv[i % MOD] % MOD;  
  49.         }  
  50. else  
  51.         {  
  52.             ans2[i] = ans2[i-1];  
  53.         }  
  54.     }  
  55. while(T--)  
  56.     {  
  57. "%d%d",&n,&m);  
  58.         LL ans = ans1[n] * ans2[m] % MOD;  
  59. "%lld\n",ans);  
  60.     }  
  61. return 0;  
  62. }  



 

接下来还有一个关于逆元的有意思的题目,描述如下


     

逆元详解_i++_70


证明:由


     

逆元详解_ios_71


     其中

逆元详解_i++_72


     所以只需要证明

逆元详解_ios_73

,而我们知道

逆元详解_#include_74


逆元详解_#include_75

的逆元对应全部     

逆元详解_#include_74

中的所有数,既是单射也是满射。


     所以进一步得到


      

逆元详解_i++_77


      证明完毕!


 



上一篇:SpringBoot 接口加密解密,新姿势!
下一篇:没有了
网友评论