作者 | 天元浪子       责编 | 欧阳姝黎

出品 | CSDN博客

围棋是全世界最古老的棋种(没有之一),也是古代哲学思想和中国传统文化的物质载体。小小纹枰,不过一尺见方,竟蕴藏着万千气象,着实令人为之着迷。少年时代的我,曾经有一段时间醉心于围棋。

标准的围棋盘由横竖各19道线组成网格,共有361个交叉点,每个交叉点上有白子、黑子和无子等三种可能的状态。那么问题来了:围棋总共有多少种不同的局面呢?

稍微思考一下,所有的程序员都会给出正确的答案:3^{361}(3的361次方)。可是,这究竟是一个多大的数字呢?算一下就知道了。

Python程序员随手写了一行代码,敲个回车,计算就结束了。

>>> pow(3,361)
174089650659031927907188238070
564367946602724950263541194828
118706801051676184649841162792
889887149386120969888163207806
137549871813550931295148033696
60572893075468180597603

C/C++程序员看完Python程序员的操作,不以为然,心里想,别看你写起来简单,速度肯定没我快。讲效率,还得看我C/C++的。

long result = 1
int i
for(i=0; i<361; i++) {result *= 3;
}

写到这里,C/C++程序员忽然意识到,long int恐怕不够用,即使long long int也只有8个字节,最大只能到2^{64}-1计算3^{361}肯定会溢出的。比long long更大的整型没有了,要是临时定义一个结构保存超大整数,再为超大整数的计算写一堆函数,恐怕一时半会儿搞不定。这可如何是好?要不用改用double float试试?赶紧上网查了一下,double可以表示-1.79E+308 ~ +1.79E+308之间的任意数,可是3^{361}3在这个范围内吗?

这时,C/C++程序员心里有点慌了。幸好有点数学功底,简单计算一下:

3^{361}大约有173位长,总算还在double覆盖的范围之内。也不用循环了,直接使用数学库中的pow函数吧。

#include <stdio.h>
#include <math.h>int main(void) {double result = pow(3,361);printf("%Lf\n", result);return 0;
}

最后,C/C++程序员给出了一个浮点类型的答案。虽然精度略有损失,但也不算离谱。我用的是CodeBlocks,显示耗时28毫秒,这里面应当包括了编译连接的时间,否则C不至于慢到这个程度。

174089650659031910000000000000
000000000000000000000000000000
000000000000000000000000000000
000000000000000000000000000000
000000000000000000000000000000
00000000000000000000000.000000Process returned 0 (0x0)
Execution time : 0.028 s

看完C/C++程序员的这番折腾,Java程序员擦擦额头的冷汗,心中暗自庆幸:多亏我大Java有BigInteger这样的神器,不然真要出丑了。

import java.math.BigInteger;BigInteger result = new BigInteger("1");
for(int i=1; i<=361; i++) {  result.multiply(new BigInteger("3")));
}

BigInteger用起来很方便,计算3^{361}毫无压力,只是不能兼容普通整型的那些运算符号,所有的运算都需要显式地调用函数,比如,这里的乘法就得调用multiply函数。

以上场景,纯属臆测,绝无褒贬任何编程语言之意,请各位明察。实际上,Python的超大整数计算也是C语言实现的,只不过采用了非常精妙的方案,最终经过各种优化,性能远超我们自己写出来的C代码。

Python的超大整数计算方案,精妙在哪儿呢?仅举存储一例:普通的Python整型采用4个字节存储,当处理超大整数时,每4个字节一个存储单元,单元之间采用2^{30}

即1073741824进制,一个单元满1073741824即向上一单元进位。

Python超大整数的存储实现

上图是超大整数1152921506754330627采用1073741824进制的存储示意图,占用了三个存储单元共计12个字节,每个单元仍然是普通的整型——这就是Python的超大整型和普通整型完全兼容的秘密。在这一点上,Python可以说完胜Java的BigInteger。不过Java还有个BigDecimal,可以无损地处理任意精度的浮点数,为Java扳回一局。

采用1073741824进制的Python的超大整数计算方案的效率如何呢?还是以计算3^{361}为例,看Python代码需要多长时间。

>>> import time
>>> def power(x, base=2):t0 = time.time()result = pow(base, x)print('耗时%.06f秒'%(time.time()-t0))return result>>> power(361, base=3)
耗时0.000000秒
174089650659031927907188238070
564367946602724950263541194828
118706801051676184649841162792
889887149386120969888163207806
137549871813550931295148033696
60572893075468180597603

太神奇了!居然连1微秒都不到?我有点怀疑这个结论,继续测试更大的数字,2的1000次方。

>>> power(1000)
耗时0.000000秒
107150860718626732094842504906
000181056140481170553360744375
038837035105112493612249319837
881569585812759467291755314682
518714528569231404359845775746
985748039345677748242309854210
746050623711418779541821530464
749835819412673987675591655439
460770629145711964776865421676
604298316526243868372056680693
76

计算2^{1000}所花时间同样少于1微秒,但是显示计算结果花费了较长时间。我把代码修改了一下,不再显示计算结果,只考察计算时间。

>>> def power(x, base=2):t0 = time.time()result = pow(base, x)print('耗时%.06f秒'%(time.time()-t0))#return result>>> power(10000) # 2的1万次方
耗时0.000000秒
>>> power(100000) # 2的10万次方
耗时0.000000秒
>>> power(1000000) # 2的100万次方
耗时0.005016秒
>>> power(10000000) # 2的1千万次方
耗时0.048000秒
>>> power(100000000) # 2的1亿次方
耗时0.620648秒
>>> power(1000000000) # 2的10亿次方
耗时7.448035秒
>>> power(10000000000) # 2的100亿次方
耗时77.881435秒

计算2的1万次方和2的10万次,所花时间仍然不足1微秒。直到计算2的100万次方时,方才显示耗时5毫秒。当算完2的100亿次方之后,我没有继续下去——2的100亿次方,这个数字实在是太过恐怖,我已经无法想象它的大小了。要知道,地球上全部物质的原子数量,也不过是1.28E47这个量级,大约是2的157次方。

那么,Python能够计算的最大整数到底有多大呢?我没有明确的概念,不过我在验证费马小定理的逆命题时,出现过一次超大整数计算错误。

a = 2
t = 2305843009213693951
s = 1152921504606846975
Traceback (most recent call last):File "huge.py", line 56, in <module>miller_rabin(x) # M61File "huge.py", line 42, in miller_rabinprint((pow(a, t*pow(2,s)) - 1)%huge_num)
MemoryError

当我试图计算pow(a, t*pow(2,s)时,发生了内存错误。这里a等于2,s大于115亿亿,t大于230亿亿。显然,这个结果远远大于2的100亿次方。


60+专家,13个技术领域,CSDN 《IT 人才成长路线图》重磅来袭!直接扫码或微信搜索「CSDN」公众号,后台回复关键词「路线图」,即可获取完整路线图!☞“时隔 10 年,重新开始写代码的我要崩溃了!”☞Google 宣布 Kotlin-first 已四年,为什么 Java 开发者仍不买账?☞“32 位应用已死!”

谁说Python慢来着?不用Python,这个问题难倒了无数的程序员相关推荐

  1. python朋友圈刷屏_“Python太火了!请救救Java!”9万程序员刷屏朋友圈 !

    没想到有生之年,笔者能观察到"霸主陨落"的过程,继PLPY4月榜单官宣,Python躺赢,再度"夺"冠,实力甩下Java和C后,近期,Stack Overflo ...

  2. Python最抢手、Go最有前途,7000位程序员揭秘2019软件开发现状

    作者 | 屠敏 整理 报告来源 | JetBrains 转载自 CSDN(ID:CSDNnews) 互联网的下半场,科技公司为面对更加严峻的竞争环境,越来越重视开源节流.而对于身处其中且撑起 IT 半 ...

  3. python图像处理实战 戴伊_这一套封面的程序员专业书籍你读过哪一本?

    以往我们总盯着畅销书,经典书,新书,今天给大家介绍Packt Publishing的程序员专业书籍.这一套封面的程序员书你读过哪一本? 1.Python图像处理实战 [印度] 桑迪潘·戴伊(Sandi ...

  4. python好学吗 老程序员-学习python,难道是为了当一名苦逼的程序员吗?

    最近在和朋友聊天的时候,朋友问了我这样一个问题,问我:你报培训班学习python有用吗?听说程序员虽然工资高,但是工作强度很大,天天加班而且还会遇到中年危机,不像律师,会计这种越老越挣钱.当我听完后, ...

  5. python里输入3.14*2会出现数据报错_Python程序员的30个常见错误

    导读:在这篇文章中,我将总结新老Python程序员常犯的一些错误,以帮助你们在自己的工作避免犯同样或类似错误. 作者:Mark Lutz 译者:伯乐在线 - 果果夫斯基 来源:http://blog. ...

  6. Go远超Python,机器学习人才极度稀缺,全球16,655位程序员告诉你这些真相

    作者 | 唐小引 题图 | HackerEarth 报告 出品 | AI科技大本营(ID:rgznai100) Go 正变得越来越受开发者欢迎. 几年前,滴滴的工程师曾告诉我,在滴滴 Go 已经得到了 ...

  7. python编程语言的缺点-常见的AI编程语言优缺点比较,程序员千万不要入错行!...

    原标题:常见的AI编程语言优缺点比较,程序员千万不要入错行! 人工智能编程是一种技术的提升,为不同公司的运营和人们的生活带来了极高的效率和最佳效益. 人工智能为不同的行业带来了另一种智能技术,其潜力的 ...

  8. Go 远超 Python,机器学习人才极度稀缺,全球 16,655 位程序员告诉你这些真相!...

    作者 | 唐小引 头图 | HackerEarth 报告 出品 | CSDN(ID:CSDNnews) Go 正变得越来越受开发者欢迎. 几年前,滴滴的工程师曾告诉我,在滴滴 Go 已经得到了非常广泛 ...

  9. Python 爬取 3 万条游戏评分数据,原来程序员最爱玩的游戏竟然是......

    作者 | 量化小白H 责编 | 胡巍巍 Python规划学习路线图,速领取? https://edu.csdn.net/topic/python115?utm_source=csdn_bw 本文爬取了 ...

最新文章

  1. (视频+图文)机器学习入门系列-第10章 人工神经网络
  2. 自动化Debias框架,一键去除推荐系统所有Bias
  3. OpenCASCADE可视化:3D演示之3D视图设施
  4. 分表后需要注意的那些事儿
  5. chrome浏览器最小字号解决方案
  6. AES加密,加签验签
  7. 微服务组件记事本:本地搭建Skywalking
  8. 谷粒商城三阶段课件_苏科版初中物理九年级上册二、变阻器公开课优质课课件教案视频...
  9. 对没有标记为安全的activex控件进行初始化和脚本运行_RASP攻防 —— RASP安全应用与局限性浅析...
  10. 只考虑用户估计的计算机时间,操作系统第四章进程调度和死锁习题及答案
  11. linux 进程内核栈
  12. 中考配额是什么意思_@所有家长!这些中考政策,你应该提前了解↘
  13. 【论文学习】Large-scale Video Classification with Convolutional Neural Networks
  14. JPA中id前面有空格导致的Column 'id' not found问题
  15. 【清华大学】操作系统 陈渝——Part7 进程/线程 管理
  16. Unity利用JsonFX解析Json格式文本
  17. 解决user installations are disabled via policy on the machine错误
  18. VR技术与生态:大咖跟你聊VR未来
  19. 常用品牌交换机端口镜像(Port Mirroring)配置
  20. EasyExcel为单个Cell设置样式

热门文章

  1. Jenkins构建步骤图解
  2. sql server 2012中red gate的sql source control消失
  3. HDU 2896 病毒侵袭(AC自动机/模板题)
  4. jsonp/ajax 自己的一些总结
  5. HDU 5515 Game of Flying Circus 二分
  6. [译] JAVA初学者的30个常见问题
  7. Java List 的merge
  8. SAS® Model Manager功能调研
  9. Hive中外部表的alter与drop操作的最低权限要求
  10. 《Linux编程》课堂测验 ·002【Shell编程】