首页 > 其他 > 详细

快速幂

时间:2021-02-14 09:39:57      阅读:30      评论:0      收藏:0      [点我收藏+]

一般而言,在计算\(a^n\)时,我们会使用暴力的方法,逐个去乘,这样操作时间复杂度是O(n)。

而是用快速幂算法时间复杂度可以降到O(logn)。

递归思想

在计算\(a^n\)时,我们可以先计算\(a^2\),然后在计算\((a^2)^2\),一直计算到\(n\)次幂。

例如\(3^9\),可以这样递归:
\(3^9=3*3^8\);
\(3^8=3^4*3^4\);
\(3^4=3^2*3^2\);
\(3^2=3^1*3^1\);
\(3^1=3^1*3^0\);
观察这个递归过程,我们可以发现:
\(n\)是偶数时,\(a^n=a^{n/2}*a^{n/2}\);
当n是奇数时,\(a^n=a*a^{n/2}\);
代码如下:

long long fastPow(long long a, long long n){
	if(n == 1) return a;
	long long ans = fastPow(a,n/2); 
	if(n % 2 == 1) return ans * ans * a; 
	else return ans * ans;
}

位运算

它是以数的二进制为基础,利用二进制的特性进行计算;

例如:计算\(2^{11}\),我们可以把它分成\(2^8\)\(2^2\)\(2^1\)的乘积,即\(2^{11}=2^8*2^2*2^1\)

那么现在可以思考三个问题:1、如何求\(2^8\)\(2^2\)\(2^1\)的值,2、如何把n分成\(11=8+2+1\),3、如何跳过不需要的部分。

1、如何求\(2^8\)\(2^2\)\(2^1\)的值
我们可以很容易发现\(2^1*2^1=2^2\)\(2^2*2^2=2^4\)\(2^4*2^4=2^8\)等等,都是2的倍数,并且都是倍乘关系,通过递推就可以实现。
2、如何把n分成\(11=8+2+1\)
用二进制我们便可以轻松理解,把n转换为二进制数,二进制数中的每一位的权值都是低一位的两倍,对应的\(a^i\)是倍乘关系,例如\(11_{10}=1011_2=2^3+2^1+2^0=8+2+1\)
3、如何跳过不需要的部分
例如\(2^{11}\),\(2^{11}=2^8*2^2*2^1\),需要跳过\(2^4\),在这里我们只需要做个判断即可,利用二进制的位运算很容易:
a、n&1,去n的最后一位,并且判断这一位是否需要跳过。
b、n>>=1,把n右移一位,去掉n的最后一位。

long long fastPow(long long a, long long n){
	long long ans = 1;
	while(n){
		if(n&1) ans *= a;
		a = a* a;
		n >>= 1;
	} 
	return ans;
}

快速幂

原文:https://www.cnblogs.com/hy2001/p/14400920.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!