總結(jié)python 三種常見的內(nèi)存泄漏場景
概要
不要以為 Python 有自動垃圾回收就不會內(nèi)存泄漏,本著它有“垃圾回收”我有“垃圾代碼”的精神,現(xiàn)在總結(jié)一下三種常見的內(nèi)存泄漏場景。
無窮大導(dǎo)致內(nèi)存泄漏
如果把內(nèi)存泄漏定義成只申請不釋放,那么借著 Python 中整數(shù)可以無窮大的這個特點,我們一行代碼就可以完成內(nèi)存泄漏了。
i = 1024 ** 1024 ** 1024
循環(huán)引用導(dǎo)致內(nèi)存泄漏
引用記數(shù)器 是 Python 垃圾回收機制的基礎(chǔ),如果一個對象的引用數(shù)量不為 0 那么是不會被垃圾回收的,我們可以通過 sys.getrefcount 來得到給定對象的引用數(shù)量。
In [1]: import sys
In [2]: a = {'name':'tom','age':16}
In [3]: sys.getrefcount(a) # 由于 getrefcount 內(nèi)部也會臨時的引用 a 所以,使得計數(shù)器的值變成了 2 。
Out[3]: 2
In [4]: b = a
In [5]: sys.getrefcount(a)
Out[5]: 3
先來看一個循環(huán)引用的場景。
#!/usr/bin/evn python3
import sys
import time
import threading
class Person(object):
free_lock = threading.Condition()
def __init__(self, name: str = ""):
"""
Parameters
----------
name: str
姓名
best_friend: str
最要好的朋友名
"""
self._name = name
self._best_friend = None
@property
def best_friend(self, person: "Person"):
return self._best_friend
@best_friend.setter
def best_friend(self, friend: "Person"):
self._best_friend = friend
def __str__(self):
"""
"""
return self._name
def __del__(self):
"""
"""
self.free_lock.acquire()
print(f"{self._name} 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。")
sys.stderr.flush()
self.free_lock.release()
def mem_leak():
"""
循環(huán)引用導(dǎo)致內(nèi)存泄漏
"""
zhang_san = Person(name='張三')
li_si = Person("李四")
# 構(gòu)造出循環(huán)引用
# 李四的好友是張三
li_si.best_friend = zhang_san
# 張三的好友是李四
zhang_san.best_friend = li_si
if __name__ == "__main__":
for i in range(3):
time.sleep(0.01)
print(f"{i}")
mem_leak()
print("mem_leak 執(zhí)行完成了.")
time.sleep(5)
運行效果。
python3 main.py
0
1
2
mem_leak 執(zhí)行完成了.
張三 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
李四 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
張三 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
李四 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
張三 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
李四 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間
由于循環(huán)引用的存在,使得 mem_leak 函數(shù)就行執(zhí)行完了其內(nèi)部的局部變量引用計數(shù)器也不為 0 ,所以內(nèi)存得不到及時的釋放。釋放這部分內(nèi)存有兩個途徑 1、 被 Python 內(nèi)部的循環(huán)檢測機制發(fā)現(xiàn)了; 2、進程退出前的集中釋放。
tracemalloc 可以在一定程序上幫我們發(fā)現(xiàn)問題,在此就不講怎么用了,我們直接上解決方案。Python 為程序員提供了弱引用,通過這種方式可以不增加對象引用計數(shù)器的數(shù)值,這成為了我們打破循環(huán)引用的一種手段。
In [1]: import sys
In [2]: import weakref
In [3]: from main import Person
In [4]: tom = Person('tom')
In [5]: sys.getrefcount(tom)
Out[5]: 2
In [6]: p = weakref.ref(tom)
In [7]: sys.getrefcount(tom) # 弱引用不會增加計數(shù)器的值
Out[7]: 2
現(xiàn)在使用 weakref 技術(shù)來改造我們的代碼。
#!/usr/bin/evn python3
import sys
import time
import weakref
import threading
class Person(object):
free_lock = threading.Condition()
def __init__(self, name: str = ""):
"""
Parameters
----------
name: str
姓名
best_friend: str
最要好的朋友名
"""
self._name = name
self._best_friend = None
@property
def best_friend(self, person: "Person"):
return self._best_friend
@best_friend.setter
def best_friend(self, friend: "Person"):
self._best_friend = weakref.ref(friend)
def __str__(self):
"""
"""
return self._name
def __del__(self):
"""
"""
self.free_lock.acquire()
print(f"{self._name} 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。")
sys.stderr.flush()
self.free_lock.release()
def mem_leak():
"""
循環(huán)引用導(dǎo)致內(nèi)存泄漏
"""
zhang_san = Person(name='張三')
li_si = Person("李四")
# 構(gòu)造出循環(huán)引用
# 李四的好友是張三
li_si.best_friend = zhang_san
# 張三的好友是李四
zhang_san.best_friend = li_si
if __name__ == "__main__":
for i in range(3):
time.sleep(0.01)
print(f"{i}")
mem_leak()
print("mem_leak 執(zhí)行完成了.")
time.sleep(5)
運行效果。
python3 main.py
0
張三 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
李四 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
1
張三 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
李四 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
2
張三 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
李四 要 GG 了,現(xiàn)在釋放它的內(nèi)存空間。
mem_leak 執(zhí)行完成了.
可以看到現(xiàn)在一旦函數(shù)執(zhí)行完成,其內(nèi)部的局部變量的內(nèi)存就會得到釋放,非常的及時。
外面庫導(dǎo)致內(nèi)存泄漏
這種情況我也只遇到過一次,之前 mysql-connector-python 的內(nèi)存泄漏,導(dǎo)致我的程序跑著跑著占用的內(nèi)存就越來越大;最后我們返的 C 語言擴展禁用之后就沒有問題了。
以上就是總結(jié)python 三種常見的內(nèi)存泄漏場景的詳細(xì)內(nèi)容,更多關(guān)于python 內(nèi)存泄漏的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Python著名游戲?qū)崙?zhàn)之方塊連接 我的世界
讀萬卷書不如行萬里路,學(xué)的扎不扎實要通過實戰(zhàn)才能看出來,本篇文章手把手帶你模仿著名游戲——我的世界,大家可以在過程中查缺補漏,看看自己掌握程度怎么樣2021-10-10
Pandas數(shù)據(jù)集的分塊讀取的實現(xiàn)
本文主要介紹了Pandas數(shù)據(jù)集的分塊讀取的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
Python3查找列表中重復(fù)元素的個數(shù)的3種方法詳解
這篇文章主要介紹了Python3查找列表中重復(fù)元素的個數(shù)方法詳解,需要的朋友可以參考下2020-02-02
Python中使用threading.Event協(xié)調(diào)線程的運行詳解
這篇文章主要介紹了Python中使用threading.Event協(xié)調(diào)線程的運行詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-05-05
Python調(diào)用百度AI實現(xiàn)圖片上表格識別功能
這篇文章主要給大家介紹了關(guān)于Python調(diào)用百度AI實現(xiàn)圖片上表格識別功能的相關(guān)資料,在Python環(huán)境下,利用百度AI開放平臺文字識別技術(shù),對表格類圖片進行識別,需要的朋友可以參考下2021-09-09

