一切皆是對象
在 Python 一切皆是對象,包括所有類型的常量與變量,整型,布爾型,甚至函數。 參見stackoverflow上的一個問題 Is everything an object in python like ruby
代碼中即可以驗證:
# everythin in python is object def fuction(): return print isinstance(True, object) print isinstance(0, object) print isinstance('a', object) print isinstance(fuction, object)
如何計算
Python 在 sys 模塊中提供函數 getsizeof 來計算 Python 對象的大小。
sys.getsizeof(object[, default]) 以字節(byte)為單位返回對象大小。 這個對象可以是任何類型的對象。 所以內置對象都能返回正確的結果 但不保證對第三方擴展有效,因為和具體實現相關。 ...... getsizeof() 調用對象的 __sizeof__ 方法, 如果對象由垃圾收集器管理, 則會加上額外的垃圾收集器開銷。
當然,對象內存占用與 Python 版本以及操作系統版本關系密切, 本文的代碼和測試結果都是基于 windows7 32位操作系統。
import sys print sys.version
2.7.2 (default, Jun 24 2011, 12:21:10) [MSC v.1500 32 bit (Intel)]
基本類型
?布爾型
print 'size of True: %d' % (sys.getsizeof(True)) print 'size of False: %d' % (sys.getsizeof(False))
輸出:
size of True: 12 size of False: 12
?整型
# normal integer print 'size of integer: %d' % (sys.getsizeof(1)) # long print 'size of long integer: %d' % (sys.getsizeof(1L)) print 'size of big long integer: %d' % (sys.getsizeof(100000L)) 輸出:
size of integer: 12x size of long integer 1L: 14 size of long integer 100000L: 16
可以看出整型占用12字節,長整型最少占用14字節,且占用空間會隨著位數的增多而變大。 在2.x版本,如果整型類型的值超出sys.maxint,則自動會擴展為長整型。而 Python 3.0 之后,整型和長整型統一為一種類型。
?浮點型
print 'size of float: %d' % (sys.getsizeof(1.0))
輸出:
size of float: 16
浮點型占用16個字節。超過一定精度后會四舍五入。
參考如下代碼:
print 1.00000000003 print 1.000000000005
輸出:
1.00000000003 1.00000000001
?字符串
# size of string type print '\r\n'.join(["size of string with %d chars: %d" % (len(elem), sys.getsizeof(elem)) for elem in ["", "a", "ab"]]) # size of unicode string print '\r\n'.join(["size of unicode string with %d chars: %d" % (len(elem), sys.getsizeof(elem)) for elem in [u"", u"a", u"ab"]])
輸出:
size of string with 0 chars: 21 size of string with 1 chars: 22 size of string with 2 chars: 23 size of unicode string with 0 chars: 26 size of unicode string with 1 chars: 28 size of unicode string with 2 chars: 30
普通空字符串占21個字節,每增加一個字符,多占用1個字節。Unicode字符串最少占用26個字節,每增加一個字符,多占用2個字節。
集合類型
?列表
# size of list type print '\r\n'.join(["size of list with %d elements: %d" % (len(elem), sys.getsizeof(elem)) for elem in [[], [0], [0,2], [0,1,2]]])
輸出:
size of list with 0 elements: 36 size of list with 1 elements: 40 size of list with 2 elements: 44 size of list with 3 elements: 48
可見列表最少占用36個字節,每增加一個元素,增加4個字節。但要注意,sys.getsizeof?函數并不計算容器類型的元素大小。比如:
print 'size of list with 3 integers %d' % (sys.getsizeof([0,1,2])) print 'size of list with 3 strings %d' % (sys.getsizeof(['0','1','2']))
輸出:
size of list with 3 integers 48 size of list with 3 strings 48
容器中保存的應該是對元素的引用。如果要準確計算容器,可以參考recursive sizeof recipe?。使用其給出的?total_size?函數:
print 'total size of list with 3 integers %d' % (total_size([0,1,2])) print 'total size of list with 3 strings %d' % (total_size(['0','1','2']))
輸出為:
total size of list with 3 integers 84 total size of list with 3 strings 114
可以看出列表的空間占用為 基本空間 36 + (對象引用 4 + 對象大小) * 元素個數。
另外還需注意如果聲明一個列表變量,則其會預先分配一些空間,以便添加元素時增加效率:
li = [] for i in range(0, 101): print 'list with %d integers size: %d, total_size: %d' % (i, getsizeof(li), total_size(li)) li.append(i)
?元組
基本與列表類似,但其最少占用為28個字節。
?字典
字典的情況相對復雜很多,具體當然要參考代碼 dictobject.c, 另外 NOTES ON OPTIMIZING DICTIONARIES 非常值得仔細閱讀。
基本情況可以參考[stackoverflow] 的問題 Python's underlying hash data structure for dictionaries 中的一些回答:
?字典最小擁有8個條目的空間(PyDict_MINSIZE);
?條目數小于50,000時,每次增長4倍;
?條目數大于50,000時,每次增長2倍;
?鍵的hash值緩存在字典中,字典調整大小后不會重新計算;
每接近2/3時,字典會調整大小。
以上這篇淺談Python 對象內存占用就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
