欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python內存泄漏和內存溢出的解決方案

 更新時間:2020年09月26日 09:06:59   作者:posted  
這篇文章主要介紹了Python內存泄漏和內存溢出的解決方案,幫助大家維護后臺進程,感興趣的朋友可以了解下

一、內存泄漏

像Java程序一樣,雖然Python本身也有垃圾回收的功能,但是同樣也會產生內存泄漏的問題。
對于一個用 python 實現(xiàn)的,長期運行的后臺服務進程來說,如果內存持續(xù)增長,那么很可能是有了“內存泄露”。

1、內存泄露的原因

對于 python 這種支持垃圾回收的語言來說,怎么還會有內存泄露? 概括來說,有以下三種原因:

  • 所用到的用 C 語言開發(fā)的底層模塊中出現(xiàn)了內存泄露。
  • 代碼中用到了全局的 list、 dict 或其它容器,不停的往這些容器中插入對象,而忘記了在使用完之后進行刪除回收
  • 代碼中有“引用循環(huán)”,并且被循環(huán)引用的對象定義了__del__方法,就會發(fā)生內存泄露。

為什么循環(huán)引用的對象定義了__del__方法后collect就不起作用了呢?

gc模塊最常使用的方法就是gc.collect()方法,使用collect方法對循環(huán)引用的對象進行垃圾回收。
如果我們在類中重載了__del__方法。__del__方法定義了在用del語句刪除對象時除了釋放內存空間以外的操作。
一般而言,在使用了del語句的時候解釋器首先會看要刪除對象的引用計數(shù),如果為0,那么就釋放內存并執(zhí)行del方法。
在這里,首先del語句出現(xiàn)時本身引用計數(shù)就不為0(因為有循環(huán)引用的存在),所以解釋器不釋放內存;
再者,執(zhí)行collect方法時應該會清除循環(huán)引用所產生的無效引用計數(shù)從而達到del的目的,對于這兩個循環(huán)引用對象而言,
python無法判斷調用它們的del方法時會不會要用到對方那個對象,比如在進行b.del()時可能會用到b._a也就是a,如果在那之前a已經被釋放,那么就徹底GG了。
為了避免這種情況,collect方法默認不對重載了del方法的循環(huán)引用對象進行回收,而它們倆的狀態(tài)也會從unreachable轉變?yōu)閡ncollectable。由于是uncollectable的,自然就不會被collect處理,所以就進入了garbage列表。

2、內存泄露的診斷思路

無論是哪種方式的內存泄露,最終表現(xiàn)的形式都是某些 python 對象在不停的增長;因此,首先是要找到這些異常的對象。

3、診斷步驟

用到的工具: gc 模塊和 objgraph 模塊

gc模塊 是Python的垃圾收集器模塊,gc使用標記清除算法回收垃圾

objgraph 是一個用于診斷內存問題的工具

  • 1、 在服務程序的循環(huán)邏輯中,選擇出一個診斷點
  • 2、 在診斷點,插入如下診斷語句  
import gc
import objgraph


### 強制進行垃圾回收 
gc.collect() 

### 打印出對象數(shù)目最多的 50 個類型信息 
objgraph.show_most_common_types(limit=50) 

4、檢查統(tǒng)計信息,找到異常對象

運行加入診斷語句的服務程序,并將打印到屏幕上的統(tǒng)計信息重定向到日志中。運行一段時間后,就可以來分析日志,看看哪些對象在不停的增長。

比如,排查結果可能是:
一個多線程程序,多個線程作為生產者,一個線程作為消費者,通過將一個 tuple 對象送入異步隊列進行通信。
由于消費者的處理速度跟不上生產者的速度,又沒有進行同步, 導致異步隊列中的對象越來越多。

二、內存溢出

1、內存溢出原因

  1. 內存中加載的數(shù)據(jù)量過于龐大,如一次從數(shù)據(jù)庫取出過多數(shù)據(jù)
  2. 集合類中有對對象的引用,使用完后未清空,產生了堆積,使得JVM不能回收
  3. 代碼中存在死循環(huán)或循環(huán)產生過多重復的對象實體
  4. 使用的第三方軟件中的BUG
  5. 啟動參數(shù)內存值設定的過小

2、內存溢出的解決方案

第一步,修改JVM啟動參數(shù),直接增加內存(-Xms,-Xmx參數(shù)一定不要忘記加)

第二步,檢查錯誤日志,查看“OutOfMemory”錯誤前是否有其 它異?;蝈e誤

第三步,對代碼進行走查和分析,找出可能發(fā)生內存溢出的位置

重點排查以下幾點:

  1. 檢查對數(shù)據(jù)庫查詢中,是否有一次獲得全部數(shù)據(jù)的查詢。一般來說,如果一次取十萬條記錄到內存,就可能引起內存溢出。這個問題比較隱蔽,在上線前,數(shù)據(jù)庫中數(shù)據(jù)較少,不容易出問題,上線后,數(shù)據(jù)庫中數(shù)據(jù)多了,一次查詢就有可能引起內存溢出。因此對于數(shù)據(jù)庫查詢盡量采用分頁的方式查詢。
  2. 檢查代碼中是否有死循環(huán)或遞歸調用。
  3. 檢查是否有大循環(huán)重復產生新對象實體。
  4. 檢查List、MAP等集合對象是否有使用完后,未清除的問題。List、MAP等集合對象會始終存有對對象的引用,使得這些對象不能被GC回收。

第四步,使用內存查看工具動態(tài)查看內存使用情況

三、內存泄漏和內存溢出的區(qū)別

內存溢出是指向JVM申請內存空間時沒有足夠的可用內存了,就會拋出OOM即內存溢出。

內存泄漏是指,向JVM申請了一塊內存空間,使用完后沒有釋放,由于沒有釋放,這塊內存區(qū)域其他類加載的時候無法申請,

同時當前類又沒有這塊內存空間的內存地址了也無法使用,相當于丟了一塊內存,這就是內存泄漏。

值得注意的是內存泄漏最終會導致內存溢出,很好理解,內存丟了很多最后當然內存不夠用了。

以上就是Python內存泄漏和內存溢出的解決方案的詳細內容,更多關于Python內存泄漏和內存溢出的資料請關注腳本之家其它相關文章!

相關文章

  • Python實現(xiàn)非正太分布的異常值檢測方式

    Python實現(xiàn)非正太分布的異常值檢測方式

    今天小編就為大家分享一篇Python實現(xiàn)非正太分布的異常值檢測方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-12-12
  • python?教程之blinker?信號庫

    python?教程之blinker?信號庫

    這篇文章主要介紹了python?教程之blinker?信號庫,文章基于python的相關資料展開詳細的內容說明。具有一定的參考價價值,需要的小伙伴可以參考一下
    2022-05-05
  • PyCharm中如何創(chuàng)建帶有注釋的py文件

    PyCharm中如何創(chuàng)建帶有注釋的py文件

    這篇文章主要介紹了 PyCharm中如何創(chuàng)建帶有注釋的py文件,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2024-08-08
  • Jupyter notebook中5個有趣的魔法命令分享

    Jupyter notebook中5個有趣的魔法命令分享

    眾?所周知,Jupyter notebook是一個交互式的Python shell,也就是IPython的封裝版,非常適合用來進行數(shù)據(jù)分析和機器學習。本文為大家整理了Jupyter notebook中5個有趣的魔法命令,感興趣的可以了解一下
    2022-07-07
  • Python如何生成隨機高斯模糊圖片詳解

    Python如何生成隨機高斯模糊圖片詳解

    這篇文章主要給大家介紹了關于高斯模糊的原理以及python實現(xiàn)的相關資料,Python使用opencv庫生成模糊圖像還是很方便的,需要的朋友可以參考下
    2021-05-05
  • Python中的各種裝飾器詳解

    Python中的各種裝飾器詳解

    這篇文章主要介紹了Python中的各種裝飾器詳解,Python裝飾器分兩部分,一是裝飾器本身的定義,一是被裝飾器對象的定義,本文分別講解了各種情況下的裝飾器,需要的朋友可以參考下
    2015-04-04
  • Python 對象序列化與反序列化之pickle json詳細解析

    Python 對象序列化與反序列化之pickle json詳細解析

    我們知道在Python中,一切皆為對象,實例是對象,類是對象,元類也是對象。本文正是要聊聊如何將這些對象有效地保存起來,以供后續(xù)使用
    2021-09-09
  • 教你學會通過python的matplotlib庫繪圖

    教你學會通過python的matplotlib庫繪圖

    今天教大家如何學會通過python的matplotlib庫繪圖,文中有非常詳細的圖文解說及代碼示例,對正在學習python的小伙伴們很有幫助,需要的朋友可以參考下
    2021-05-05
  • 深入分析Python中Lambda函數(shù)的用法

    深入分析Python中Lambda函數(shù)的用法

    lambda函數(shù)是Python中常用的內置函數(shù),又稱為匿名函數(shù)。和普通函數(shù)相比,它只有函數(shù)體,省略了def和return,使得結構看起來更精簡。本文將詳細說說Lambda函數(shù)的用法,需要的可以參考一下
    2022-12-12
  • Python一步步帶你操作Excel

    Python一步步帶你操作Excel

    這篇文章主要介紹了Python編寫命令行腳本操作excel的方法,文章圍繞主題展開詳細的內容介紹,具有一定的參考價值,需要的朋友可以參考一下
    2022-08-08

最新評論