今天就跟大家聊聊有關(guān)Python中怎么優(yōu)化占用的內(nèi)存,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
成都創(chuàng)新互聯(lián)公司主要從事成都做網(wǎng)站、成都網(wǎng)站制作、網(wǎng)頁(yè)設(shè)計(jì)、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)屏南,10多年網(wǎng)站建設(shè)經(jīng)驗(yàn),價(jià)格優(yōu)惠、服務(wù)專(zhuān)業(yè),歡迎來(lái)電咨詢(xún)建站服務(wù):18980820575使用Python內(nèi)置的數(shù)據(jù)結(jié)構(gòu)Dict來(lái)實(shí)現(xiàn)上述例子的需求很簡(jiǎn)單。
>>> ob = {'x':1, 'y':2, 'z':3} >>> x = ob['x'] >>> ob['y'] = y
查看以下ob這個(gè)對(duì)象占用的內(nèi)存大?。?/p>
>>> print(sys.getsizeof(ob)) 240
簡(jiǎn)單的三個(gè)整數(shù),占用的內(nèi)存還真不少,想象以下,如果有大量的這樣的數(shù)據(jù)要存儲(chǔ),會(huì)占用更大的內(nèi)存。
對(duì)于喜歡面向?qū)ο缶幊痰某绦騿T來(lái)說(shuō),更喜歡把數(shù)據(jù)包在一個(gè)class里。使用class使用同樣需求:
class Point: # def __init__(self, x, y, z): self.x = x self.y = y self.z = z >>> ob = Point(1,2,3)
class的數(shù)據(jù)結(jié)構(gòu)和Dict區(qū)別就很大了,我們來(lái)看看這種情況下占用內(nèi)存的情況:
關(guān)于 _weakref_(弱引用)可以查看這個(gè)文檔, 對(duì)象的__dict__中存儲(chǔ)了一些self.xxx的一些東西。從Python 3.3開(kāi)始,key使用了共享內(nèi)存存儲(chǔ), 減少了RAM中實(shí)例跟蹤的大小。
>>> print(sys.getsizeof(ob), sys.getsizeof(ob.__dict__)) 56 112
可以看到內(nèi)存占用量,class比dict少了一些,但這遠(yuǎn)遠(yuǎn)不夠。
從class的內(nèi)存占用分布上,我們可以發(fā)現(xiàn),通過(guò)消除__dict__和_weakref__,可以顯著減少RAM中類(lèi)實(shí)例的大小,我們可以通過(guò)使用__slots__來(lái)達(dá)到這個(gè)目的。
class Point: __slots__ = 'x', 'y', 'z' def __init__(self, x, y, z): self.x = x self.y = y self.z = z >>> ob = Point(1,2,3) >>> print(sys.getsizeof(ob)) 64
可以看到內(nèi)存占用顯著的減少了
默認(rèn)情況下,Python的新式類(lèi)和經(jīng)典類(lèi)的實(shí)例都有一個(gè)dict來(lái)存儲(chǔ)實(shí)例的屬性。這在一般情況下還不錯(cuò),而且非常靈活,乃至在程序中可以隨意設(shè)置新的屬性。但是,對(duì)一些在”編譯”前就知道有幾個(gè)固定屬性的小class來(lái)說(shuō),這個(gè)dict就有點(diǎn)浪費(fèi)內(nèi)存了。
當(dāng)需要?jiǎng)?chuàng)建大量實(shí)例的時(shí)候,這個(gè)問(wèn)題變得尤為突出。一種解決方法是在新式類(lèi)中定義一個(gè)__slots__屬性。
__slots__聲明中包含若干實(shí)例變量,并為每個(gè)實(shí)例預(yù)留恰好足夠的空間來(lái)保存每個(gè)變量;這樣Python就不會(huì)再使用dict,從而節(jié)省空間。
那么用slot就是非非常那個(gè)有必要嗎?使用__slots__也是有副作用的:
每個(gè)繼承的子類(lèi)都要重新定義一遍_(kāi)_slots__
實(shí)例只能包含哪些在__slots__定義的屬性,這對(duì)寫(xiě)程序的靈活性有影響,比如你由于某個(gè)原因新網(wǎng)給instance設(shè)置一個(gè)新的屬性,比如instance.a = 1, 但是由于a不在__slots__里面就直接報(bào)錯(cuò)了,你得不斷地去修改__slots__或者用其他方法迂回的解決
實(shí)例不能有弱引用(weakref)目標(biāo),否則要記得把__weakref__放進(jìn)__slots__
最后,namedlist和attrs提供了自動(dòng)創(chuàng)建帶__slot__的類(lèi),感興趣的可以試試看。
Python還有一個(gè)內(nèi)置類(lèi)型元組,用于表示不可變數(shù)據(jù)結(jié)構(gòu)。 元組是固定的結(jié)構(gòu)或記錄,但沒(méi)有字段名稱(chēng)。 對(duì)于字段訪(fǎng)問(wèn),使用字段索引。 在創(chuàng)建元組實(shí)例時(shí),元組字段一次性與值對(duì)象關(guān)聯(lián):
>>> ob = (1,2,3) >>> x = ob[0] >>> ob[1] = y # ERROR
元組的示例很簡(jiǎn)潔:
>>> print(sys.getsizeof(ob)) 72
可以看只比__slot__多8byte:
通過(guò)namedtuple我們也可以實(shí)現(xiàn)通過(guò)key值來(lái)訪(fǎng)問(wèn)tuple里的元素:
Point = namedtuple('Point', ('x', 'y', 'z'))
它創(chuàng)建了一個(gè)元組的子類(lèi),其中定義了用于按名稱(chēng)訪(fǎng)問(wèn)字段的描述符。 對(duì)于我們的例子,它看起來(lái)像這樣:
class Point(tuple): # @property def _get_x(self): return self[0] @property def _get_y(self): return self[1] @property def _get_y(self): return self[2] # def __new__(cls, x, y, z): return tuple.__new__(cls, (x, y, z))
此類(lèi)的所有實(shí)例都具有與元組相同的內(nèi)存占用。 大量實(shí)例會(huì)留下稍大的內(nèi)存占用:
python的第三方庫(kù)recordclassd提供了一個(gè)數(shù)據(jù)結(jié)構(gòu)recordclass.mutabletuple,它幾乎和內(nèi)置tuple數(shù)據(jù)結(jié)構(gòu)一致,但是占用更少的內(nèi)存。
>>> Point = recordclass('Point', ('x', 'y', 'z')) >>> ob = Point(1, 2, 3)
實(shí)例化以后,只少了PyGC_Head:
到此,我們可以看到,和__slot__比,又進(jìn)一步縮小了內(nèi)存占用:
recordclass提供了另外一個(gè)解決方法:在內(nèi)存中使用與__slots__類(lèi)相同的存儲(chǔ)結(jié)構(gòu),但不參與循環(huán)垃圾收集機(jī)制。通過(guò)recordclass.make_dataclass可以創(chuàng)建出這樣的實(shí)例:
>>> Point = make_dataclass('Point', ('x', 'y', 'z'))
另外一個(gè)方法是繼承自dataobject
class Point(dataobject): x:int y:int z:int
以這種方式創(chuàng)建的類(lèi)將創(chuàng)建不參與循環(huán)垃圾收集機(jī)制的實(shí)例。 內(nèi)存中實(shí)例的結(jié)構(gòu)與__slots__的情況相同,但沒(méi)有PyGC_Head:
>>> ob = Point(1,2,3) >>> print(sys.getsizeof(ob)) 40
要訪(fǎng)問(wèn)這些字段,還使用特殊描述符通過(guò)其從對(duì)象開(kāi)頭的偏移量來(lái)訪(fǎng)問(wèn)字段,這些對(duì)象位于類(lèi)字典中:
mappingproxy({'__new__': <staticmethod at 0x7f203c4e6be0>, ....................................... 'x': <recordclass.dataobject.dataslotgetset at 0x7f203c55c690>, 'y': <recordclass.dataobject.dataslotgetset at 0x7f203c55c670>, 'z': <recordclass.dataobject.dataslotgetset at 0x7f203c55c410>})
有一種方法基于Cython的使用。 它的優(yōu)點(diǎn)是字段可以采用C語(yǔ)言原子類(lèi)型的值。例如:
cdef class Python: cdef public int x, y, z def __init__(self, x, y, z): self.x = x self.y = y self.z = z
這種情況下,占用的內(nèi)存更?。?/p>
>>> ob = Point(1,2,3) >>> print(sys.getsizeof(ob)) 32
內(nèi)存結(jié)構(gòu)分布如下:
但是,從Python代碼訪(fǎng)問(wèn)時(shí),每次都會(huì)執(zhí)行從int到Python對(duì)象的轉(zhuǎn)換,反之亦然。
在純Python的環(huán)境中,使用Numpy能帶來(lái)更好的效果,例如:
>>> Point = numpy.dtype(('x', numpy.int32), ('y', numpy.int32), ('z', numpy.int32)])
創(chuàng)建初始值是0的數(shù)組:
>>> points = numpy.zeros(N, dtype=Point)
看完上述內(nèi)容,你們對(duì)Python中怎么優(yōu)化占用的內(nèi)存有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注創(chuàng)新互聯(lián)-成都網(wǎng)站建設(shè)公司行業(yè)資訊頻道,感謝大家的支持。
分享文章:Python中怎么優(yōu)化占用的內(nèi)存-創(chuàng)新互聯(lián)
當(dāng)前網(wǎng)址:http://muchs.cn/article2/poiic.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供面包屑導(dǎo)航、網(wǎng)站維護(hù)、動(dòng)態(tài)網(wǎng)站、搜索引擎優(yōu)化、域名注冊(cè)、微信小程序
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶(hù)投稿、用戶(hù)轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話(huà):028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容