本文始发于个人公众号:TechFlow,原创不易,求个关注

今天是LeetCode的第31篇文章,我们来看下LeetCode的第50题,求一个数的幂。

题意

这道题的题意只有一句话,就是给定两个数x和n,要求 x n x^n xn。

从题意来看,这道题平平无奇,基本上没有什么特别的。但是我们继续看它的note就会发现问题,其中x是浮点数,它的范围是-100到100。而n的范围则是32位int的范围,到这里就有问题了。

因为如果n很大,我们用循环去计算 x n x^n xn的话一定会超时。我们之前讨论过这个问题,即使是运算最快的C++,一秒种内的运算次数也只有10的8次方左右。而32位的int高达10的9次方,如果是Python的话这个运算会更慢,所以我们用循环肯定是无法得出结果的。

快速幂

这道题目把复杂度限制死了,我们从暴力方法入手是想不出结果的,必须要引入新的算法。而针对本题的算法就是快速幂

快速幂算法本质上也是二进制算法的变形,需要基于我们对二进制的充分理解,某种程度上来说它和多重背包问题当中的二进制拆分的解法比较近似。都需要对整数进行拆分,不过不同的是在背包问题当中拆分的结果进行的累加运算,而这里则是累乘。

如果你没有看过多重背包问题,也没有关系,我会从头开始将它讲清楚。

快速幂的算法好理解,但是看懂了很容易忘,尤其是初学的时候。学的时候理解了,过两天就忘了,或者代码写不出来了,这很正常。所以我们先从根本问题出发,从根源上理解它,而不只是运作原理。

第一个问题:我们使用快速幂的原因是什么?

这个问题很好回答,当然是因为啊,不然的话我们用循环计算幂不行么。但是为什么快速幂就快呢?为什么它比循环快呢?

这个问题哪怕我们没有学过快速幂的算法,也是可以回答的,它的答案也很简单,因为我们把一个原本不太好求的量转化成了一个很容易求解的量,从而降低了复杂度。说起来是废话的东西,但其实是很多算法的本质。算法也不是天上掉下来的,想出算法的人也不是凭空拍脑袋就出来的,最核心都是有一个原理可寻的。很多算法的核心原理,其实就是用转化和代替的方法求本来不是特别方便求的值。这个方法不仅在数学上常用,在算法上也是一样。

我们理解了这个核心之后,剩下的就简单了,我们知道 x n x^n xn不好求,因为我们现在没什么好的办法,那什么量是容易求的量呢?又该怎么转化呢?顺着这个思路,我们眼前的世界清楚了很多。

原本我们求解 x n x^n xn的方法就是通过循环,每次循环都乘上一个x,循环n次之后得到结果。我们观察一下这个过程,会发现我们在循环的时候,每一次循环,其实都代表了x的指数增加了1。也就是说它是线性增长的,当然就慢了。那什么增长比较快呢?指数增长比较快,比如我们一直翻倍翻倍,就很快。有一个关于指数增长的著名故事,说是从前有一个人发明了国际象棋。当地的国王非常喜欢下棋,于是他就把这个发明者召到了面前来说,你这个发明很好,我愿意奖赏你,你说吧,你想要什么,只要我能给的都行。

这个人说,陛下我想要的很简单,只要一些米。我希望第一个格子里放1颗米,第二个格子里放2颗,第3个格子里放4颗。每一个格子里放的都是前面的2倍,请陛下把这些米赏赐给国家里的穷人吧。

国王很震惊,觉得这个人是不是缺心眼,这么好的机会怎么只要了这么点赏赐。但是我们都知道,国际象棋一共有八八六十四格,我们按照这样放满,需要 2 65 − 1 2^{65}-1 265−1颗米。这显然是一个天文数字,别说一个国家了,就是全世界的米加起来也没这么多。故事里没提这个人最后的结局,很有可能因为戏弄国王被砍死了。但不管结局如何,至少说明了一个问题,指数增长和我们的直觉不符,它的变化极快

所以我们希望要是我们的指数也可以这样成倍地增长而不是每次只增加1就好了,那么我们怎么让指数成倍增长呢?这个就很简单了,平方就好了

我们把x平方就得到了 x 2 x^2 x2,我们再把它平方就得到了 x 4 x^4 x4,我们每次平方完,指数都翻了一倍,就好像刚才故事里的棋盘一样。所以我们只需要很少的次数就可以让指数变得很大。

我们解决了指数增长速度的问题,但是又遇到了新的问题,我们这样增长是很快,但是它翻倍再翻倍不一定就能得到n呀?

这个问题也简单,直接得到不可能就想办法呗。举个例子,比如我们的n=15,我们先找到小于15的最大的2的幂,发现是8。所以我们先得到了 x 8 x^8 x8,我们把它放在一边之后还剩下了7,我们再来凑7,又找到了4,于是再放到一边,还剩下3,我们再来凑3……如此循环往复直到凑齐了n为止,n是一定可以这样凑到的,因为这本质上就是一个转化成二进制的过程。

我把整个过程画成了一张图,我们来看下图:

我们先算出所有2的幂,然后在算出所有x的2的幂次方。再把n拆成二进制,把二进制当中对应位置是1的值乘起来,就得到了结果。

有些同学可能不太熟悉二进制和位运算,我会提供两个版本的代码,帮助大家理解。我们先来看简单一些的代码:

class Solution:def myPow(self, x: float, n: int) -> float:base = []# 标记n是否为负数flag = n < 0# 把n置为正数n = abs(n)base.append(x)# 我们算出所有的x^2^ifor i in range(32):base.append(base[-1] * base[-1])ret = 1.0# 遍历32个二进制位# 如果i位为1,那么答案乘上base[i]for i in range(32):if ((1 << i) & n) > 0:ret *= base[i]# 如果n为负数返回1/retreturn 1/ret if flag else ret

在上面这段代码当中我们是先算出了每一个 x 2 i x^{2^i} x2i,然后我们再根据n转化成二进制的结果将它们乘在了一起。当然,我们熟练了之后是可以不这么费劲的,我们可以把这两步合并在一起,我们再来看一段代码:

class Solution:def myPow(self, x: float, n: int) -> float:base = []flag = n < 0n = abs(n)base = x    ret = 1.0# 我们在遍历二进制位的时候顺便求出x^2^ifor i in range(32):if ((1 << i) & n) > 0:ret *= base# x^2^i = x^2^(i-1) * x^2^(i-1)base *= basereturn 1/ret if flag else ret

总结

到这里关于快速幂的讲解就结束了,我个人感觉应该还是比较清楚的,算法的核心根基还是二进制,如果对二进制的概念掌握了,快速幂算法就是小意思了。

可能你会觉得奇怪,为什么非要用二进制而不是三进制、四进制呢,不是更快吗?原因有两个,一个是计算机底层基于二进制,我们暂时没有找到一个很好的材料可以实现三进制,因为二进制很简单,逻辑门开和关,带来高低电平表示状态就好了,但是三个状态用什么材料来实现呢?第二个原因是没有必要,多进制的确会更快,但是很有限,所以没有这个必要。

说来也不怕笑话,我在刚开始入门的时候,一直是用上的上面那种比较蠢的方法。而且放眼题解,至今没有找到一个人用这种蠢办法写快速幂的。但是代码蠢没有关系,能够运行,能够AC,能够理解才是关键。一开始写蠢点的代码没有关系,只要保持进步,以后自然会越来越好的。

今天的文章就是这些,如果觉得有所收获,请顺手点个关注或者转发吧,你们的举手之劳对我来说很重要。

LeetCode50——一题学会快速幂算法相关推荐

  1. 快速幂算法相关题目(Leetcode题解-Python语言)

    50. Pow(x, n) 快速幂算法的目的,就是快速计算 x 的 n 次方.基本思路是把 n 视作二进制数,则 n 可以被分解为多个 2 的幂次方之和,如 12 对应 1100 等于 0∗20+0∗ ...

  2. java位运算求幂,程序员必学:快速幂算法

    前阵子,有小伙伴在我B站的算法教程底下留言 小伙伴们有任何疑问或者希望我解说任何内容,都可以在我的小我私家B站或民众号(xmg_mj)留言哦,我会尽我最大能力.只管抽时间去写文章\录视频来回应人人. ...

  3. 你必须掌握的,快速幂算法

    文章目录 1. 使用快速幂算法实现Pow(x,n) 2. 快速幂取余 1. 使用快速幂算法实现Pow(x,n) 求x的n次方,可以使用暴力解法,这种算法时间复杂度为O(n),并且,当x和n比较大的时候 ...

  4. RSA密码的手动算法+快速幂算法

    公钥加密方案(非对称加密算法) 1.为什么要引入公钥加密方案? 密码学中的加密方案分成对称密钥和非对称密钥(也就是我们说的公钥加密,代表加密算法是RSA加密算法). 而对称加密方法有一个特点,任何通信 ...

  5. 快速幂算法 超详细教程

    快速幂 求幂运算 快速幂引入 快速幂 二进制 快速幂 指数折半 快速幂的应用 求幂运算 求幂运算大家都不陌生,幂是指数运算的结果,当m是正整数时nᵐ的意义为m个n相乘,n的m次幂也就是n的m次方.用代 ...

  6. 快速幂算法的原理及实现

    幂运算,即次方运算,例如计算 的值即是幂运算,在实现的时候我们往往是这样写的: int __pow(int a,int b){int ans = 1;while(b--){ans *= a;}retu ...

  7. leetcode算法总结 —— 快速幂算法

    文章目录 1. 引出快速幂算法 2. 简化语句 3. 使用位运算来提升性能 4. 对应leetcode题型 参考的是大神的文章,这篇文章相当好 https://blog.csdn.net/qq_197 ...

  8. (快速幂算法+高精度)洛谷P1045 麦森数

    前言   故事的最后,让我们以一道十分经典的题目--<麦森数>来结尾.接受现实吧,总会有我们没准备过的高精度运算出现.我们固然可以提前把高精度的快速幂模板也准备好,但是总会有百密一疏的时候 ...

  9. 六十八、快速幂算法、牛顿迭代法、累加数组+二分查找的变形

    @Author:Runsen 编程的本质来源于算法,而算法的本质来源于数学,编程只不过将数学题进行代码化. ---- Runsen 上次介绍了二分查找算法及其四个变形问题,下面介绍二分法常用的场景和典 ...

最新文章

  1. 【Android 内存优化】Bitmap 硬盘缓存 ( Google 官方 Bitmap 示例 | DiskLruCache 开源库 | 代码示例 )
  2. OpenCV 升降维度
  3. linux-security-limits
  4. esp8266make相关文件改进
  5. 自定义Redis序列化工具
  6. Linux最终将会领先于Windows、Mac OS!
  7. 悦虎144固件,华强北二代悦虎144固件,1562M芯片144固件
  8. mysql sql优化入门_Mysql入门SQL 语句优化方法30例
  9. 第2关:子节点创建、列出、删除
  10. 查看服务器是有有默认共享文件,服务器共享文件远程查看
  11. 前端—每天5道面试题(1)
  12. Scala笔记2——IDE配置、函数式编程核心概念
  13. 创作原创歌词的韵律十三辙与韵脚押韵方法
  14. Windows 自带硬盘修复命令 CHKDSK
  15. java特别描述错误的是,关于javac命令作用的描述错误的是
  16. 暴力电脑锁机生成器(加机械硬盘锁)
  17. python如何将数组里的数提取出来_python [:3] 实现提取数组中的数
  18. 360浏览器怎么导入html,360浏览器收藏夹导入/导出方法详解
  19. 心电图实验(使用vivado进行编程,VHDL语言)
  20. SpringCloud SpringBoot uniapp vue b2b2c 微服务 多商家入驻直播带货商城 分销商城 秒杀 高并发电商之责任链模式

热门文章

  1. chrome android 导航,将 Chrome for Android 的地址栏移动到屏幕下方[Android]
  2. Python使用traceback.print_exc()输出异常信息
  3. Solidity学习笔记
  4. arcgis 矢量编辑过程时,防止误移动操作设置粘滞移动容差设置大一点
  5. android的A/B到底是什么?OTA升级又是什么?
  6. SSM框架报错分析(一)——There is no getter for property named 'XXX' in 'class java.lang.String'...
  7. lr1分析器c语言实验报告怎么写,编译原理课程的设计构造LR分析法语法分析器.doc...
  8. android模拟器发送短信
  9. iOS小工具合集-(合一Kit)
  10. 如何通过命令提示符进入MySQL服务器