內存溢出和內存泄露的區別

情景描述:
有一次被問到了內存溢出和內存泄露的區別是啥?只知道內存溢出會報錯OOM(OutOfMemoryError),內存溢出Memory leak,但你說裏面的原由和還真是不知道的,很尷尬,去看帖子

內存溢出

內存溢出是指在申請內存時,沒有足夠的內存空間供使用,即系統已經不能再分配出你所需要的空間,比如你需要100M的空間,系統只剩90M了,這就叫內存溢出OutofMemeryError
例子:一個盤子用盡各種方法只能裝4個果子,你裝了5個,結果掉倒地上不能吃了。這就是溢出。比方說棧,棧滿時再做進棧必定產生空間溢出,叫上溢,棧空時再做退棧也產生空間溢出,稱爲下溢。就是分配的內存不足以放下數據項序列,稱爲內存溢出。說白了就是我承受不了那麼多,那我就報錯。
再如:給了你一塊存儲int類型數據的存儲空間,但是你卻存儲long類型的數據,那麼結果就是內存不夠用,此時就會報錯OOM。

內存泄露

內存泄露(Memory Leak)內存泄漏是指你向系統申請分配內存進行使用(new),可是使用完了以後卻不歸還(delete),結果你申請到的那塊內存你自己也不能再訪問(也許你把它的地址給弄丟了),而系統也不能再次將它分配給需要的程序。強引用所指向的對象不會被回收,可能導致內存泄漏,虛擬機寧願拋出OOM也不會去回收他指向的對象,還有代碼缺陷導致對象不斷創建而垃圾回收無法及時回收對象,或者根本不回收,比如創建線程再不用後不關閉,頻繁創建對象而不銷燬等,長時間運行下來,再大的內存都會OutOfMemoryError,可以這麼理解,恩怨是積少成多的,遲早會爆發就形成了OOM。

以發生的方式來分類:

內存溢出的原因及解決方法

內存溢出原因:

有這些內存區域會導致內存溢出

  • 虛擬機棧報出內存溢出: 如果 Java 虛擬機棧可以動態擴展(當前大部分的java虛擬機都可動態擴展,只不過Java虛擬機棧規範中也允許固定長度的虛擬機棧),並且擴展的動作已經嘗試過,但是目前無法申請到足夠的內存去完成擴展,或者在建立新的線程時沒有足夠的內存去創建對應的虛擬機棧,那 Java虛擬機將會拋出一個OutOfMemoryError 異常。
  • 本地方法棧報出內存溢出: 如果本地方法棧可以動態擴展, 並且擴展的動作已經嘗試過,但是目前無法申請到足夠內存去完成擴展,或者在建立新的線程時沒有足夠的內存去創建對應本地方法棧,那Java虛擬機將會拋出一個OutofMemoryError異常。
  • Java堆報出內存溢出:Java堆可以處於物理不連續的內存空間中,只要邏輯上是連續的即可,就像我們磁盤空間一樣,在實現時,既可以實現成固定大小的,也可以是擴展的,不過當前主流的虛擬機都是按照可以擴展實現的(通過 -Xmx和-Xms控制)如果在堆中沒有內存完成實例分配,並堆也無法再擴展時,就會拋出OutOfMemoryError異常
  • 方法區報出內存溢出:如果方法區的內存空間不能滿足內存分配請求,那 Java 虛擬機將拋出一個OutOfMemoryError 異常。
  • 運行時常量池報出內存溢出: 當創建類和接口的時候,如果構造運行時常量池所需要的內存空間超過了方法區所能提供的最大值,那Java 虛擬機將會拋出一個OutofMemoryError異常
  • 直接內存報出內存溢出: 本機直接內存的分配不會受到Java堆大小的限制,但是,既然是內存,肯定還是會受到本機總內存(包括RAM以及SWAP區或者分頁文件)大小以及處理器尋址空間的限制。服務器管理員在配置虛擬機參數時,會根據實際內存設置-Xmx等參數信息,但經常忽略直接內存,使得各個內存區域總和大於物理內存限制(包括物理的和操作系統級別的限制)從而導致動態擴展時出現OutOfMemoryError異常。

具體實際應用場景

  1. 內存中加載的數據量過於龐大,如一次從數據庫取出過多數據;
  2. 集合類中有對對象的引用,使用完後未清空,使得JVM不能回收;
  3. 代碼中存在死循環或循環產生過多重複的對象實體;
  4. 使用的第三方軟件中的BUG;
  5. 啓動參數內存值設定的過小

內存溢出的原因及解決方法

  • 修改JVM啓動參數,直接增加內存。(-Xms,-Xmx參數一定不要忘記加)
  • 檢查錯誤日誌,查看“OutOfMemory”錯誤前是否有其 它異常或錯誤
  • 對代碼進行走查和分析,找出可能發生內存溢出的位置
  • 使用內存查看工具動態查看內存使用情況

對代碼分析找出可能發生內存溢出的位置, 可能出現的幾種情況:

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

內存泄露原因

  1. 常發性內存泄漏。發生內存泄漏的代碼會被多次執行到,每次被執行的時候都會導致一塊內存泄漏。
  2. 偶發性內存泄漏。發生內存泄漏的代碼只有在某些特定環境或操作過程下才會發生。常發性和偶發性是相對的。對於特定的環境,偶發性的也許就變成了常發性的。所以測試環境和測試方法對檢測內存泄漏至關重要。
  3. 一次性內存泄漏。發生內存泄漏的代碼只會被執行一次,或者由於算法上的缺陷,導致總會有一塊僅且一塊內存發生泄漏。比如,在類的構造函數中分配內存,在析構函數中卻沒有釋放該內存,所以內存泄漏只會發生一次。
  4. 隱式內存泄漏。程序在運行過程中不停的分配內存,但是直到結束的時候才釋放內存。嚴格的說這裏並沒有發生內存泄漏,因爲最終程序釋放了所有申請的內存。但是對於一個服務器程序,需要運行幾天,幾周甚至幾個月,不及時釋放內存也可能導致最終耗盡系統的所有內存。所以,我們稱這類內存泄漏爲隱式內存泄漏。
    從用戶使用程序的角度來看,內存泄漏本身不會產生什麼危害,作爲一般的用戶,根本感覺不到內存泄漏的存在。真正有危害的是內存泄漏的堆積,這會最終消耗盡系統所有的內存。從這個角度來說,一次性內存泄漏並沒有什麼危害,因爲它不會堆積,而隱式內存泄漏危害性則非常大,因爲較之於常發性和偶發性內存泄漏它更難被檢測到

兩者區別

內存溢出代碼本身沒有原則問題,最多算是代碼品質不高,想要消除報錯只能增加java分配的內存或者優化代碼、而內存泄漏則是由於代碼存在缺陷,如果不修改代碼問題,即使分配再大的內存也會最終報錯

如果你看到了這裏請你一部看下這位大佬的文章,這位大佬寫的更全面https://www.cnblogs.com/dyh004/p/8296958.html

參考:
https://blog.csdn.net/buutterfly/article/details/6617375
https://www.cnblogs.com/rgever/p/8899758.html
https://blog.csdn.net/ruiruihahaha/article/details/70270574

發佈了14 篇原創文章 · 獲贊 10 · 訪問量 2萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章