快速幂/快速幂取模/矩阵求解快速幂

举报
Linux猿 发表于 2021/08/05 23:45:17 2021/08/05
【摘要】 因为做了一个Fibonacci的题需要用矩阵+快速幂求解所以引发了下面一系列问题!!! 快速幂:             快速幂时间复杂度为 O(log2N), 与朴素的O(N)相比效率有了极大的提高。     &nbsp...

因为做了一个Fibonacci的题需要用矩阵+快速幂求解所以引发了下面一系列问题!!!

快速幂:

            快速幂时间复杂度为 O(log2N), 与朴素的O(N)相比效率有了极大的提高。

            例如:

                       3 ^ 999 = 3 * 3 * 3 * … * 3 
                       直接乘要做998次乘法。但事实上可以这样做: 
                                                            3 ^ 2 = 3 * 3 
                                                            3 ^ 4 = (3 ^ 2) * (3 ^ 2) 
                                                            …………
                                                            3 ^ 256 = (3 ^ 128) * (3 ^ 128) 
                                                            3 ^ 512 = (3 ^ 256) * (3 ^ 256) 
                                                            再相乘: 
                                                                        3 ^ 999 
                                                                     = 3 ^ (512 + 256 + 128 + 64 + 32 + 4 + 2 + 1) 
                                                                     = (3 ^ 512) * (3 ^ 256) * (3 ^ 128) * (3 ^ 64) * (3 ^ 32) * (3 ^ 4) * (3 ^ 2) * 3 
                                     这样只要做16次乘法。
          代码中k%2的意思是如果当前这一位的二进制是 1 还是 0 ,如果是 1 ,则表示将其乘 ans (即加入最终结果),否则不乘 ans (即不加入最终结果)。

 代码:


  
  1. #include <stdio.h>
  2. int power(int n, int k)
  3. {
  4. int ans = 1;
  5. while( k )
  6. {
  7. if(k&1)
  8. ans *= n ;
  9. k= k/2;
  10. n *= n;
  11. }
  12. return ans ;
  13. }
  14. int main ( )
  15. {
  16. int n, k;
  17. while(scanf("%d %d",&n, &k)!=EOF)
  18. {
  19. printf("%d\n",power(n,k));
  20. }
  21. }

快速幂取模:

       快速幂取模还不算太懂,只知道需要对每一个值取模,公式 a*b%c=((a%c)*b)%c

代码:


  
  1. #include<stdio.h>
  2. int main()
  3. {
  4. long long a,b,c,d;
  5. int T;
  6. scanf("%d",&T);
  7. while(T--)
  8. {
  9. scanf("%lld%lld%lld",&a,&b,&c);
  10. long long ans=1;
  11. while( b )
  12. {
  13. if( b%2 )
  14. ans = ( ans * a ) % c;
  15. b/=2;
  16. a = ( a * a ) % c;
  17. }
  18. printf("%lld\n",ans);
  19. }
  20. return 0;
  21. }


矩阵+快速幂 求解Fibonacci数列:

           矩阵:矩阵可以看成一个n×m的数表,用二维数组表示
           矩阵乘法:定义矩阵A,B。A和B可以乘法操作当且仅当A的大小是a×b,B的大小是b×c,设矩阵C=AB,则C的大小是a×c,

        且有:

   

            最普通的矩阵乘法是直接三个for循环直接计算而已,所以复杂度是O(n3).

        

            
  运用矩阵乘法快速幂,可以快速计算出矩阵B^(n-1),这样实现将时间复杂度降低到O(log n).

代码(nyoj 698题 A Coin Problem 别人优代码 ):


  
  1. #include<stdio.h>
  2. #include<iostream>
  3. #include<algorithm>
  4. #include<string>
  5. #include<cstring>
  6. #include<map>
  7. #include<vector>
  8. using namespace std;
  9. const int Max = 10000;
  10. struct M
  11. {
  12. long long p[2][2];
  13. };
  14. M mult(M a,M b)
  15. {
  16. int i,j,k;
  17. M c;
  18. for(i=0;i<2;i++)
  19. for(j=0;j<2;j++)
  20. {
  21. c.p[i][j]=0;
  22. for(k=0;k<2;k++)
  23. {
  24. c.p[i][j]=(c.p[i][j]+a.p[i][k]*b.p[k][j]%Max)%Max;
  25. }
  26. }
  27. return c;
  28. }
  29. M pow(M a,long long k)
  30. {
  31. M b={1,0,0,1};
  32. while(k)
  33. {
  34. if(k&1)
  35. b=mult(b,a);
  36. a=mult(a,a);
  37. k>>=1;
  38. }
  39. return b;
  40. }
  41. int main()
  42. {
  43. int t;
  44. long long n;
  45. scanf("%d",&t);
  46. while(t--)
  47. {
  48. scanf("%lld",&n);
  49. M a={1,1,1,0};
  50. M c;
  51. c=pow(a,n-1);
  52. printf("%lld\n",(c.p[0][0]*2+c.p[0][1]*1)%Max);
  53. }
  54. return 0;
  55. }


还不明白请点击~>
 

文章来源: blog.csdn.net,作者:Linux猿,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/nyist_zxp/article/details/9335585

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。