日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > python >内容正文

python

为什么Python中整型不会溢出

發(fā)布時間:2023/12/13 python 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 为什么Python中整型不会溢出 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

前言

本次分析基于 CPython 解釋器,python3.x版本

在python2時代,整型有?int?類型和?long?長整型,長整型不存在溢出問題,即可以存放任意大小的整數(shù)。在python3后,統(tǒng)一使用了長整型。這也是吸引科研人員的一部分了,適合大數(shù)據(jù)運算,不會溢出,也不會有其他語言那樣還分短整型,整型,長整型...因此python就降低其他行業(yè)的學習門檻了。

那么,不溢出的整型實現(xiàn)上是否可行呢?

不溢出的整型的可行性

盡管在 C 語言中,整型所表示的大小是有范圍的,但是 python 代碼是保存到文本文件中的,也就是說,python代碼中并不是一下子就轉(zhuǎn)化成 C 語言的整型的,我們需要重新定義一種數(shù)據(jù)結(jié)構(gòu)來表示和存儲我們新的“整型”。

怎么來存儲呢,既然我們要表示任意大小,那就得用動態(tài)的可變長的結(jié)構(gòu),顯然,數(shù)組的形式能夠勝任:

[longintrepr.h] struct _longobject {PyObject_VAR_HEADint *ob_digit; };

?

長整型的保存形式

長整型在python內(nèi)部是用一個?int?數(shù)組(?ob_digit[n]?)保存值的. 待存儲的數(shù)值的低位信息放于低位下標, 高位信息放于高下標.比如要保存?123456789?較大的數(shù)字,但我們的int只能保存3位(假設):

ob_digit[0] = 789; ob_digit[1] = 456; ob_digit[2] = 123;

低索引保存的是地位,那么每個?int?元素保存多大的數(shù)合適?有同學會認為數(shù)組中每個int存放它的上限(2^31 - 1),這樣表示大數(shù)時,數(shù)組長度更短,更省空間。但是,空間確實是更省了,但操作會代碼麻煩,比方大數(shù)做乘積操作,由于元素之間存在乘法溢出問題,又得多考慮一種溢出的情況。

怎么來改進呢?在長整型的?ob_digit?中元素理論上可以保存的int類型有?32?位,但是我們只保存?15?位,這樣元素之間的乘積就可以只用?int?類型保存即可, 結(jié)果做位移操作就能得到尾部和進位?carry?了,定義位移長度為?15:

#define PyLong_SHIFT 15 #define PyLong_BASE ((digit)1 << PyLong_SHIFT) #define PyLong_MASK ((digit)(PyLong_BASE - 1))

PyLong_MASK?也就是?0b111111111111111?,通過與它做位運算?與?的操作就能得到低位數(shù)。

有了這種存放方式,在內(nèi)存空間允許的情況下,我們就可以存放任意大小的數(shù)字了。

?

長整型的運算

加法與乘法運算都可以使用我們小學的豎式計算方法,例如對于加法運算:

??ob_digit[2]ob_digit[1]ob_digit[0]
加數(shù)a?23934543
加數(shù)b+?454632
結(jié)果z?24389175

為方便理解,表格展示的是數(shù)組中每個元素保存的是 3 位十進制數(shù),計算結(jié)果保存在變量z中,那么 z 的數(shù)組最多只要?size_a + 1?的空間(兩個加數(shù)中數(shù)組較大的元素個數(shù) + 1),因此對于加法運算,可以這樣來處理:

[longobject.c] static PyLongObject * x_add(PyLongObject *a, PyLongObject *b) {int size_a = len(a), size_b = len(b);PyLongObject *z;int i;int carry = 0; // 進位// 確保a是兩個加數(shù)中較大的一個if (size_a < size_b) {// 交換兩個加數(shù)swap(a, b);swap(&size_a, &size_b);}z = _PyLong_New(size_a + 1); // 申請一個能容納size_a+1個元素的長整型對象for (i = 0; i < size_b; ++i) {carry += a->ob_digit[i] + b->ob_digit[i];z->ob_digit[i] = carry & PyLong_MASK; // 掩碼carry >>= PyLong_SHIFT; // 移除低15位, 得到進位}for (; i < size_a; ++i) { // 單獨處理a中高位數(shù)字carry += a->ob_digit[i];z->ob_digit[i] = carry & PyLong_MASK;carry >>= PyLong_SHIFT;}z->ob_digit[i] = carry;return long_normalize(z); // 整理元素個數(shù)}

這部分的過程就是,先將兩個加數(shù)中長度較長的作為第一個加數(shù),再為用于保存結(jié)果的 z 申請空間,兩個加數(shù)從數(shù)組從低位向高位計算,處理結(jié)果的進位,將結(jié)果的低 15 位賦值給 z 相應的位置。最后的?long_normalize(z)是一個整理函數(shù),因為我們 z 申請了?a_size + 1?的空間,但不意味著 z 會全部用到,因此這個函數(shù)會做一些調(diào)整,去掉多余的空間,數(shù)組長度調(diào)整至正確的數(shù)量,若不方便理解,附錄將給出更利于理解的python代碼。

豎式計算不是按個位十位來計算的嗎,為什么這邊用整個元素?

豎式計算方法適用與任何進制的數(shù)字,我們可以這樣來理解,這是一個 32768 (2的15次方) 進制的,那么就可以把數(shù)組索引為 0 的元素當做是 “個位”,索引 1 的元素當做是 “十位”。

乘法運算

乘法運算一樣可以用豎式的計算方式,兩個乘數(shù)相乘,存放結(jié)果的 z 的元素個數(shù)為?size_a + size_b?即可:

?操作??ob_digit[2]ob_digit[1]ob_digit[0]
乘數(shù)a???23934543
乘數(shù)b*???454632
結(jié)果z??15126631176
??10866282522?
結(jié)果z?10881409153176

這里需要主意的是,當乘數(shù) b 用索引 i 的元素進行計算時,結(jié)果 z 也是從 i 索引開始保存。先創(chuàng)建 z 并初始化為 0,這 z 上做累加操作,加法運算則可以利用前面的?x_add?函數(shù):

// 為方便理解,會與cpython中源碼部分稍有不同 static PyLongObject * x_mul(PyLongObject *a, PyLongObject *b) {int size_a = len(a), size_b = len(b);PyLongObject *z = _PyLong_New(size_a + size_b);memset(z->ob_digit, 0, len(z) * sizeof(int)); // z 的數(shù)組清 0for (i = 0; i < size_b; ++i) {int carry = 0; // 用一個int保存元素之間的乘法結(jié)果int f = b->ob_digit[i]; // 當前乘數(shù)b的元素// 創(chuàng)建一個臨時變量,保存當前元素的計算結(jié)果,用于累加PyLongObject *temp = _PyLong_New(size_a + size_b);memset(temp->ob_digit, 0, len(temp) * sizeof(int)); // temp 的數(shù)組清 0int pz = i; // 存放到臨時變量的低位for (j = 0; j < size_a; ++j) {carry = f * a[j] + carry;temp[pz] = carry & PyLong_MASK; // 取低15位carry = carry >> PyLong_SHIFT; // 保留進位pz ++;}if (carry){ // 處理進位carry += temp[pz];temp[pz] = carry & PyLong_MASK;carry = carry >> PyLong_SHIFT;}if (carry){temp[pz] += carry & PyLong_MASK;}temp = long_normalize(temp);z = x_add(z, temp);}return z}

這大致就是乘法的處理過程,豎式乘法的復雜度是n^2,當數(shù)字非常大的時候(數(shù)組元素個數(shù)超過 70 個)時,python會選擇性能更好,更高效的?Karatsuba multiplication?乘法運算方式,這種的算法復雜度是 3nlog3≈3n1.585,當然這種計算方法已經(jīng)不是今天討論的內(nèi)容了。有興趣的小伙伴可以去了解下。

總結(jié)

要想支持任意大小的整數(shù)運算,首先要找到適合存放整數(shù)的方式,本篇介紹了用 int 數(shù)組來存放,當然也可以用字符串來存儲。找到合適的數(shù)據(jù)結(jié)構(gòu)后,要重新定義整型的所有運算操作,本篇雖然只介紹了加法和乘法的處理過程,但其實還需要做很多的工作諸如減法,除法,位運算,取模,取余等。

python代碼以文本形式存放,因此最后,還需要一個將字符串形式的數(shù)字轉(zhuǎn)換成這種整型結(jié)構(gòu):

[longobject.c] PyObject * PyLong_FromString(const char *str, char **pend, int base) { }

這部分不是本篇的重點,有興趣的同學可以看看這個轉(zhuǎn)換的過程。

參考

  • longobject.c

附錄

# 例子中的表格中,數(shù)組元素最多存放3位整數(shù),因此這邊設置1000 # 對應的取低位與取高位也就變成對 1000 取模和取余操作 PyLong_SHIFT = 1000 PyLong_MASK = 999# 以15位長度的二進制 # PyLong_SHIFT = 15 # PyLong_MASK = (1 << 15) - 1def long_normalize(num):"""去掉多余的空間,調(diào)整數(shù)組的到正確的長度eg: [176, 631, 0, 0] ==> [176, 631]:param num::return:"""end = len(num)while end >= 1:if num[end - 1] != 0:breakend -= 1num = num[:end]return numdef x_add(a, b):size_a = len(a)size_b = len(b)carry = 0# 確保 a 是兩個加數(shù)較大的,較大指的是元素的個數(shù)if size_a < size_b:size_a, size_b = size_b, size_aa, b = b, az = [0] * (size_a + 1)i = 0while i < size_b:carry += a[i] + b[i]z[i] = carry % PyLong_SHIFTcarry //= PyLong_SHIFTi += 1while i < size_a:carry += a[i]z[i] = carry % PyLong_SHIFTcarry //= PyLong_SHIFTi += 1z[i] = carry# 去掉多余的空間,數(shù)組長度調(diào)整至正確的數(shù)量z = long_normalize(z)return zdef x_mul(a, b):size_a = len(a)size_b = len(b)z = [0] * (size_a + size_b)for i in range(size_b):carry = 0f = b[i]# 創(chuàng)建一個臨時變量temp = [0] * (size_a + size_b)pz = ifor j in range(size_a):carry += f * a[j]temp[pz] = carry % PyLong_SHIFTcarry //= PyLong_SHIFTpz += 1if carry: # 處理進位carry += temp[pz]temp[pz] = carry % PyLong_SHIFTcarry //= PyLong_SHIFTpz += 1if carry:temp[pz] += carry % PyLong_SHIFTtemp = long_normalize(temp)z = x_add(z, temp) # 累加return za = [543, 934, 23] b = [632, 454] print(x_add(a, b)) print(x_mul(a, b))

?

本文由?hongweipeng?創(chuàng)作

總結(jié)

以上是生活随笔為你收集整理的为什么Python中整型不会溢出的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。