zoukankan      html  css  js  c++  java
  • 通过程序了解快速幂和模取幂运算的优化

    建议先看第三个有解释的程序。

    快速幂a^b

     1 #include <stdio.h>
     2 #include <stdlib.h>
     3 
     4 int main()
     5 {
     6     long a,b,result=1;
     7     //a^b
     8     scanf("%ld%ld",&a,&b);
     9     while (b)
    10     {
    11         if ((b & 1)==1)
    12             result=result*a;
    13         a=a*a;
    14         b>>=1;
    15     }
    16     printf("%ld
    ",result);
    17     return 0;
    18 }

    求a^b,当a固定,b有很多种不同的取值时,可以用这种奇葩但挺高效的方法:

     1 #include <stdio.h>
     2 #include <stdlib.h>
     3 #include <math.h>
     4 #define max 2147483647
     5 
     6 int main()
     7 {
     8     long a,b,s,i,x[32],y[32];
     9     //a^b
    10     scanf("%ld",&a);
    11     if (a==1)
    12     {
    13         while (scanf("%ld",&b)!=EOF)
    14             printf("1
    ");
    15         return 0;
    16     }
    17     x[0]=a;
    18     y[0]=1;
    19     //a^t=max t=loga(max)=log(max)/log(a)
    20     //2^i=t i=log2(t)=log(t)/log(2)
    21     for (i=1;i<=(long)(log(log(max)/log(a))/log(2));i++)
    22     {
    23         x[i]=x[i-1]*x[i-1];
    24         y[i]=y[i-1]<<1;
    25     }
    26     while (scanf("%ld",&b)!=EOF)
    27     {
    28         s=1;
    29         while (b)
    30         {
    31             //求数字为1的最高位
    32             i=(long)log(b)/log(2);
    33             s*=x[i];
    34             b-=y[i];
    35         }
    36         printf("%ld
    ",s);
    37     }
    38     return 0;
    39 }

    快速幂求模取幂运算a^b mod c

     1 #include <stdio.h>
     2 #include <stdlib.h>
     3 
     4 int main()
     5 {
     6     //a^b mod c
     7     long a,b,c,result=1;
     8     scanf("%ld%ld%ld",&a,&b,&c);
     9     while (b)
    10     {
    11         if ((b &1)==1)
    12             result=result*a%c;
    13         a=a*a%c;
    14         b=b>>1;
    15     }
    16     printf("%ld
    ",result);
    17     return 0;
    18 }
    19 /*
    20 Input:12996 227 37909
    21 Output:7775
    22 */

     

    1.b=b>>1为b除以2,也指二进制下b去掉最后1位

    2.a=a*a,执行k次,a(现在)=a(原来)^(2^k)

    3.当while循环执行到第k+1次时,当前b的最后一位为原来b的倒数第k位,当该位为1,result乘上a(原来)^(2^k),即为现在a的值

    当该位为0,不进行操作。

    4.b=2^n*t(n)+2^(n-1)*t(n-1)+…+2^1*t(1)+2^0*t(0)

    b以二进制表示,t(r)的值为0或1,r=n,n-1,…,0

    5.a^b=a^(2^n)*t(n)  *  a^(2^(n-1))*t(n-1)  * … *  a^1*t(1)  *  a^0*t(0)

           当t(r)为0,该项值为1,可省去

    相当于3中所说,当倒数第k位值为1时,结果乘上a^(2^k)

     

     

    注意:用位运算,比加减乘除运算速度快很多,所以b=b/2可以改为b=b>>1

     

    用法:

    1.优化:欧拉函数

    2.求结果的末k位,只有加法,减法和乘法

    优化:中间结果只保留末k位,因为第k位以前的内容对结果的末k位没有影响

  • 相关阅读:
    bat文件转换为exe文件
    桌面快捷方式增(删)
    客户推广微信小程序的几种方法如下
    小程序开发客户对接流程
    Java MySQL 连接
    前端开发构建工具
    [转]chrome 的devtools 中setting 开启workspace , 也有点用处。不是很大
    回归基础的东西,不能只是“感觉会了”
    angular 的ui.router 定义不同的state 对应相同的url
    学习javascript 非常好的博客
  • 原文地址:https://www.cnblogs.com/cmyg/p/6759900.html
Copyright © 2011-2022 走看看