溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Python占用內存如何優化

發布時間:2021-09-06 16:57:06 來源:億速云 閱讀:214 作者:小新 欄目:開發技術

這篇文章主要為大家展示了“Python占用內存如何優化”,內容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“Python占用內存如何優化”這篇文章吧。

舉個栗子

我們舉個簡單的場景,使用Python存儲一個三維坐標數據,x,y,z。

Dict

使用Python內置的數據結構Dict來實現上述例子的需求很簡單。

>>> ob = {'x':1, 'y':2, 'z':3}
>>> x = ob['x']
>>> ob['y'] = y

查看以下ob這個對象占用的內存大?。?/p>

>>> print(sys.getsizeof(ob))
240

簡單的三個整數,占用的內存還真不少,想象以下,如果有大量的這樣的數據要存儲,會占用更大的內存。

數據量占用內存大小
1 000 000240 Mb
10 000 0002.40 Gb
100 000 00024 Gb

Class

對于喜歡面向對象編程的程序員來說,更喜歡把數據包在一個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的數據結構和Dict區別就很大了,我們來看看這種情況下占用內存的情況:

字段占用內存
PyGC_Head24
PyObject_HEAD16
__weakref__8
__dict__8
TOTAL56

關于 __weakref__(弱引用)可以查看這個文檔, 對象的dict中存儲了一些self.xxx的一些東西。從Python 3.3開始,key使用了共享內存存儲, 減少了RAM中實例跟蹤的大小。

>>> print(sys.getsizeof(ob), sys.getsizeof(ob.__dict__)) 
56 112
數據量占用內存
1 000 000168 Mb
10 000 0001.68 Gb
100 000 00016.8 Gb

可以看到內存占用量,class比dict少了一些,但這遠遠不夠。

__slots__

從class的內存占用分布上,我們可以發現,通過消除dict和_weakref__,可以顯著減少RAM中類實例的大小,我們可以通過使用slots來達到這個目的。

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

可以看到內存占用顯著的減少了

字段內存占用
PyGC_Head24
PyObject_HEAD16
x8
y8
z8
TOTAL64
數據量占用內存
1 000 00064Mb
10 000 000640Mb
100 000 0006.4Gb

默認情況下,Python的新式類和經典類的實例都有一個dict來存儲實例的屬性。這在一般情況下還不錯,而且非常靈活,乃至在程序中可以隨意設置新的屬性。但是,對一些在”編譯”前就知道有幾個固定屬性的小class來說,這個dict就有點浪費內存了。

當需要創建大量實例的時候,這個問題變得尤為突出。一種解決方法是在新式類中定義一個slots屬性。

slots聲明中包含若干實例變量,并為每個實例預留恰好足夠的空間來保存每個變量;這樣Python就不會再使用dict,從而節省空間。

那么用slot就是非非常那個有必要嗎?使用slots也是有副作用的:

  1. 每個繼承的子類都要重新定義一遍slots

  2. 實例只能包含哪些在slots定義的屬性,這對寫程序的靈活性有影響,比如你由于某個原因新網給instance設置一個新的屬性,比如instance.a = 1, 但是由于a不在slots里面就直接報錯了,你得不斷地去修改slots或者用其他方法迂回的解決

  3. 實例不能有弱引用(weakref)目標,否則要記得把weakref放進slots

最后,namedlist和attrs提供了自動創建帶slot的類,感興趣的可以試試看。

Tuple

Python還有一個內置類型元組,用于表示不可變數據結構。 元組是固定的結構或記錄,但沒有字段名稱。 對于字段訪問,使用字段索引。 在創建元組實例時,元組字段一次性與值對象關聯:

>>> ob = (1,2,3)
>>> x = ob[0]
>>> ob[1] = y # ERROR

元組的示例很簡潔:

>>> print(sys.getsizeof(ob))
72

可以看只比slot多8byte:

字段占用內存(bytes)
PyGC_Head24
PyObject_HEAD16
ob_size8
[0]8
[1]8
[2]8
TOTAL72

Namedtuple

通過namedtuple我們也可以實現通過key值來訪問tuple里的元素:

Point = namedtuple('Point', ('x', 'y', 'z'))

它創建了一個元組的子類,其中定義了用于按名稱訪問字段的描述符。 對于我們的例子,它看起來像這樣:

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))

此類的所有實例都具有與元組相同的內存占用。 大量實例會留下稍大的內存占用:

數據量內存占用
1 000 00072 Mb
10 000 000720 Mb
100 000 0007.2 Gb

Recordclass

python的第三方庫recordclassd提供了一個數據結構recordclass.mutabletuple,它幾乎和內置tuple數據結構一致,但是占用更少的內存。

>>> Point = recordclass('Point', ('x', 'y', 'z'))
>>> ob = Point(1, 2, 3)

實例化以后,只少了PyGC_Head:

字段占用內存
PyObject_HEAD16
ob_size8
x8
y8
y8
TOTAL48

到此,我們可以看到,和slot比,又進一步縮小了內存占用:

數據量內存占用
1 000 00048 Mb
10 000 000480 Mb
100 000 0004.8 Gb

Dataobject

recordclass提供了另外一個解決方法:在內存中使用與slots類相同的存儲結構,但不參與循環垃圾收集機制。通過recordclass.make_dataclass可以創建出這樣的實例:

>>> Point = make_dataclass('Point', ('x', 'y', 'z'))

另外一個方法是繼承自dataobject

class Point(dataobject):
 x:int
 y:int
 z:int

以這種方式創建的類將創建不參與循環垃圾收集機制的實例。 內存中實例的結構與slots的情況相同,但沒有PyGC_Head:

字段內存占用(bytes)
PyObject_HEAD16
x8
y8
y8
TOTAL40
>>> ob = Point(1,2,3)
>>> print(sys.getsizeof(ob))
40

要訪問這些字段,還使用特殊描述符通過其從對象開頭的偏移量來訪問字段,這些對象位于類字典中:

mappingproxy({'__new__': <staticmethod at 0x7f203c4e6be0>,
    .......................................
    'x': <recordclass.dataobject.dataslotgetset at 0x7f203c55c690>,
    'y': <recordclass.dataobject.dataslotgetset at 0x7f203c55c670>,
    'z': <recordclass.dataobject.dataslotgetset at 0x7f203c55c410>})
數據量內存占用
1 000 00040 Mb
10 000 000400 Mb
100 000 0004.0 Gb

Cython

有一種方法基于Cython的使用。 它的優點是字段可以采用C語言原子類型的值。例如:

cdef class Python:
 cdef public int x, y, z

 def __init__(self, x, y, z):
  self.x = x
  self.y = y
  self.z = z

這種情況下,占用的內存更?。?/p>

>>> ob = Point(1,2,3)
>>> print(sys.getsizeof(ob))
32

內存結構分布如下:

字段內存占用(bytes)
PyObject_HEAD16
x4
y4
y4
пусто4
TOTAL32
數據量內存占用
1 000 00032 Mb
10 000 000320 Mb
100 000 0003.2 Gb

但是,從Python代碼訪問時,每次都會執行從int到Python對象的轉換,反之亦然。

Numpy

在純Python的環境中,使用Numpy能帶來更好的效果,例如:

>>> Point = numpy.dtype(('x', numpy.int32), ('y', numpy.int32), ('z', numpy.int32)])

創建初始值是0的數組:

>>> points = numpy.zeros(N, dtype=Point)
數據量內存占用
1 000 00012 Mb
10 000 000120 Mb
100 000 0001.2 Gb

最后

可以看出,在Python性能優化這方面,還是有很多事情可以做的。Python提供了方便的同時,也需要暫用較多的資源。在不通的場景下,我需要選擇不同的處理方法,以便帶來更好的性能體驗。

以上是“Python占用內存如何優化”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女