日韩久久久精品,亚洲精品久久久久久久久久久,亚洲欧美一区二区三区国产精品 ,一区二区福利

Python 的整數(shù)與 Numpy 的數(shù)據(jù)溢出

系統(tǒng) 1990 0

某位 A 同學(xué)發(fā)了我一張截圖,問為何結(jié)果中出現(xiàn)了負(fù)數(shù)?

Python 的整數(shù)與 Numpy 的數(shù)據(jù)溢出_第1張圖片

看了圖,我第一感覺就是數(shù)據(jù)溢出了。數(shù)據(jù)超出能表示的最大值,就會(huì)出現(xiàn)奇奇怪怪的結(jié)果。

然后,他繼續(xù)發(fā)了張圖,內(nèi)容是 print(100000*208378),就是直接打印上圖的 E[0]*G[0],結(jié)果是 20837800000,這是個(gè)正確的結(jié)果。

所以新的問題是:如果說上圖的數(shù)據(jù)溢出了,為何直接相乘的數(shù)卻沒有溢出?

由于我一直忽視數(shù)據(jù)的表示規(guī)則(整型的上限是多少?),而且對(duì) Numpy 了解不多,還錯(cuò)看了圖中結(jié)果,誤以為每一個(gè)數(shù)據(jù)都是錯(cuò)誤的,所以就解答不出來。

最后,經(jīng)過學(xué)習(xí)群里的一番討論,我才終于明白是怎么回事,所以本文把相關(guān)知識(shí)點(diǎn)做個(gè)梳理。

在正式開始之前,先總結(jié)一下上圖會(huì)引出的話題:

  • Python 3 中整數(shù)的上限是多少?Python 2 呢?
  • Numpy 中整數(shù)的上限是多少?出現(xiàn)整數(shù)溢出該怎么辦?

關(guān)于第一個(gè)問題,先看看 Python 2,它有兩種整數(shù):

  • 一種是短整數(shù),也即常說的整數(shù),用 int 表示,有個(gè)內(nèi)置函數(shù) int()。其大小有限,可通過 sys.maxint() 查看(取決于平臺(tái)是 32 位還是 64 位)
  • 一種是長(zhǎng)整數(shù),即大小無限的整數(shù),用 long 表示,有個(gè)內(nèi)置函數(shù) long()。寫法上是在數(shù)字后面加大寫字母 L 或小寫的 l,如 1000L

當(dāng)一個(gè)整數(shù)超出短整數(shù)范圍時(shí),它會(huì)自動(dòng)采用長(zhǎng)整數(shù)表示。舉例,打印 2**100 ,結(jié)果會(huì)在末尾加字母 L 表示它是長(zhǎng)整數(shù)。

但是到了 Python 3,情況就不同了:它僅有一種內(nèi)置的整數(shù),表示為 int,形式上是 Python 2 的短整數(shù),但實(shí)際上它能表示的范圍無限,行為上更像是長(zhǎng)整數(shù)。無論多大的數(shù),結(jié)尾都不需要字母 L 來作區(qū)分。

也就是說,Python 3 整合了兩種整數(shù)表示法,用戶不再需要自行區(qū)分,全交給底層按需處理。

理論上,Python 3 中的整數(shù)沒有上限(只要不超出內(nèi)存空間)。這就解釋了前文中直接打印兩數(shù)相乘,為什么結(jié)果會(huì)正確了。

PEP-237(Unifying Long Integers and Integers)中對(duì)這個(gè)轉(zhuǎn)變作了說明。它解釋這樣做的 目的:

這會(huì)給新的 Python 程序員(無論他們是否是編程新手)減少一項(xiàng)上手前要學(xué)的功課。

Python 在語(yǔ)言運(yùn)用層屏蔽了很多瑣碎的活,比如內(nèi)存分配,所以,我們?cè)谑褂米址⒘斜砘蜃值涞葘?duì)象時(shí),根本不用操心。整數(shù)類型的轉(zhuǎn)變,也是出于這樣的便利目的。(壞處是犧牲了一些效率,在此就不談了)

回到前面的第二個(gè)話題:Numpy 中整數(shù)的上限是多少?

由于它是 C 語(yǔ)言實(shí)現(xiàn),在整數(shù)表示上,用的是 C 語(yǔ)言的規(guī)則,也就是會(huì)區(qū)分整數(shù)和長(zhǎng)整數(shù)。

有一種方式可查看:

          
            import numpy as np

a = np.arange(2)
type(a[0])

# 結(jié)果:numpy.int32
          
        

也就是說它默認(rèn)的整數(shù) int 是 32 位,表示范圍在 -2147483648 ~ 2147483647。

對(duì)照前文的截圖,里面只有兩組數(shù)字相乘時(shí)沒有溢出:100007*4549、100012*13264,其它數(shù)據(jù)組都溢出了,所以出現(xiàn)奇怪的負(fù)數(shù)結(jié)果。

Numpy 支持的數(shù)據(jù)類型要比 Python 的多,相互間的區(qū)分界限很多樣:

Python 的整數(shù)與 Numpy 的數(shù)據(jù)溢出_第2張圖片

截圖來源:https://www.runoob.com/numpy/numpy-dtype.html

要解決整數(shù)溢出問題,可以通過指定 dtype 的方式:

          
            import numpy as np

q = [100000]
w = [500000]

# 一個(gè)溢出的例子:
a = np.array(q)
b = np.array(w)
print(a*b)  # 產(chǎn)生溢出,結(jié)果是個(gè)奇怪的數(shù)值

# 一個(gè)解決的例子:
c = np.array(q, dtype='int64')
d = np.array(w, dtype='int64')
print(c*d) # 沒有溢出:[50000000000]
          
        

好了,前面提出的問題就回答完了。來作個(gè)結(jié)尾吧:

  • Python 3 極大地簡(jiǎn)化了整數(shù)的表示,效果可表述為:整數(shù)就只有一種整數(shù)(int),沒有其它類型的整數(shù)(long、int8、int64 之類的)
  • Numpy 中的整數(shù)類型對(duì)應(yīng)于 C 語(yǔ)言的數(shù)據(jù)類型,每種“整數(shù)”有自己的區(qū)間,要解決數(shù)據(jù)溢出問題,需要指定更大的數(shù)據(jù)類型(dtype)

Python 的整數(shù)與 Numpy 的數(shù)據(jù)溢出_第3張圖片

公眾號(hào)【 Python貓 】, 本號(hào)連載優(yōu)質(zhì)的系列文章,有喵星哲學(xué)貓系列、Python進(jìn)階系列、好書推薦系列、技術(shù)寫作、優(yōu)質(zhì)英文推薦與翻譯等等,歡迎關(guān)注哦。


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號(hào)聯(lián)系: 360901061

您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對(duì)您有幫助就好】

您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對(duì)您有幫助,請(qǐng)用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長(zhǎng)會(huì)非常 感謝您的哦!!!

發(fā)表我的評(píng)論
最新評(píng)論 總共0條評(píng)論
主站蜘蛛池模板: 三亚市| 石台县| 高雄市| 庄河市| 望谟县| 龙江县| 清徐县| 德格县| 湛江市| 喀喇沁旗| 高州市| 新巴尔虎左旗| 鲜城| 安宁市| 通辽市| 个旧市| 柘荣县| 黔东| 莱芜市| 皮山县| 会理县| 宜兰县| 论坛| 偏关县| 蓬莱市| 花莲市| 宁海县| 七台河市| 永州市| 疏勒县| 民和| 静宁县| 山阴县| 博乐市| 文安县| 石林| 福鼎市| 翁牛特旗| 神农架林区| 库尔勒市| 耒阳市|