亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

總結python 三種常見的內存泄漏場景

瀏覽:24日期:2022-07-04 17:40:37

概要

不要以為 Python 有自動垃圾回收就不會內存泄漏,本著它有“垃圾回收”我有“垃圾代碼”的精神,現在總結一下三種常見的內存泄漏場景。

無窮大導致內存泄漏

如果把內存泄漏定義成只申請不釋放,那么借著 Python 中整數可以無窮大的這個特點,我們一行代碼就可以完成內存泄漏了。

i = 1024 ** 1024 ** 1024

循環引用導致內存泄漏

引用記數器 是 Python 垃圾回收機制的基礎,如果一個對象的引用數量不為 0 那么是不會被垃圾回收的,我們可以通過 sys.getrefcount 來得到給定對象的引用數量。

In [1]: import sys In [2]: a = {’name’:’tom’,’age’:16} In [3]: sys.getrefcount(a) # 由于 getrefcount 內部也會臨時的引用 a 所以,使得計數器的值變成了 2 。 Out[3]: 2In [4]: b = a In [5]: sys.getrefcount(a) Out[5]: 3

先來看一個循環引用的場景。

#!/usr/bin/evn python3import sysimport timeimport threadingclass 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 了,現在釋放它的內存空間。') sys.stderr.flush() self.free_lock.release()def mem_leak(): ''' 循環引用導致內存泄漏 ''' zhang_san = Person(name=’張三’) li_si = Person('李四') # 構造出循環引用 # 李四的好友是張三 li_si.best_friend = zhang_san # 張三的好友是李四 zhang_san.best_friend = li_siif __name__ == '__main__': for i in range(3): time.sleep(0.01) print(f'{i}') mem_leak() print('mem_leak 執行完成了.') time.sleep(5)

運行效果。

python3 main.py 012mem_leak 執行完成了.張三 要 GG 了,現在釋放它的內存空間。李四 要 GG 了,現在釋放它的內存空間。張三 要 GG 了,現在釋放它的內存空間。李四 要 GG 了,現在釋放它的內存空間。張三 要 GG 了,現在釋放它的內存空間。李四 要 GG 了,現在釋放它的內存空間

由于循環引用的存在,使得 mem_leak 函數就行執行完了其內部的局部變量引用計數器也不為 0 ,所以內存得不到及時的釋放。釋放這部分內存有兩個途徑 1、 被 Python 內部的循環檢測機制發現了; 2、進程退出前的集中釋放。

tracemalloc 可以在一定程序上幫我們發現問題,在此就不講怎么用了,我們直接上解決方案。Python 為程序員提供了弱引用,通過這種方式可以不增加對象引用計數器的數值,這成為了我們打破循環引用的一種手段。

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]: 2In [6]: p = weakref.ref(tom) In [7]: sys.getrefcount(tom) # 弱引用不會增加計數器的值Out[7]: 2

現在使用 weakref 技術來改造我們的代碼。

#!/usr/bin/evn python3import sysimport timeimport weakrefimport threadingclass 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 了,現在釋放它的內存空間。') sys.stderr.flush() self.free_lock.release()def mem_leak(): ''' 循環引用導致內存泄漏 ''' zhang_san = Person(name=’張三’) li_si = Person('李四') # 構造出循環引用 # 李四的好友是張三 li_si.best_friend = zhang_san # 張三的好友是李四 zhang_san.best_friend = li_siif __name__ == '__main__': for i in range(3): time.sleep(0.01) print(f'{i}') mem_leak() print('mem_leak 執行完成了.') time.sleep(5)

運行效果。

python3 main.py 0張三 要 GG 了,現在釋放它的內存空間。李四 要 GG 了,現在釋放它的內存空間。1張三 要 GG 了,現在釋放它的內存空間。李四 要 GG 了,現在釋放它的內存空間。2張三 要 GG 了,現在釋放它的內存空間。李四 要 GG 了,現在釋放它的內存空間。mem_leak 執行完成了.

可以看到現在一旦函數執行完成,其內部的局部變量的內存就會得到釋放,非常的及時。

外面庫導致內存泄漏

這種情況我也只遇到過一次,之前 mysql-connector-python 的內存泄漏,導致我的程序跑著跑著占用的內存就越來越大;最后我們返的 C 語言擴展禁用之后就沒有問題了。

以上就是總結python 三種常見的內存泄漏場景的詳細內容,更多關于python 內存泄漏的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 日韩在线视频免费不卡一区 | 国产精品久久久久9999高清 | 亚洲国产精久久小蝌蚪 | 在线精品国精品国产不卡 | 91精品日本久久久久久牛牛 | 真人毛片免费全部播放完整 | 色欧美与xxxxx | 欧美午夜a级限制福利片 | 大片免费看大片费看大片 | 亚洲国产精品自在在线观看 | 人人干人人玩 | 超级碰碰碰碰97久久久久 | 日韩3级| 国产美女做爰免费视 | 国产欧美在线不卡 | 欧美一级a俄罗斯毛片 | 黄色片中文 | 在线观看国产精品一区 | 欧美亚洲另类久久综合 | 精品视频一区二区三区在线播放 | 三级黄色在线 | 亚洲午夜在线观看 | 深爱激情四射 | 看的免费污污网站 | 黄色美女一级片 | 免费看黄色一级片 | 久久成人免费大片 | 免费的黄色网址 | 欧美一区二区三区在线观看免费 | 色综合久久中文色婷婷 | 亚洲加勒比久久88色综合一区 | 亚洲午夜久久久久影院 | 亚洲 欧美 自拍 另类 欧美 | 日本黄色美女视频 | 日本一本在线视频 | 国内精品免费视频 | 亚洲精品国产一区二区 | 超乳w真性中出し冲田杏梨101 | 女性一级全黄生活片在线播放 | 免费黄色在线观看 | 狠狠色香婷婷久久亚洲精品 |