天天看點

java jvm 記憶體溢出和記憶體洩漏的差別

1、概念的區分

記憶體溢出 out of memory

是指程式在申請記憶體時,沒有足夠的記憶體空間供其使用,出現 out of memory 。

比如申請了一個10MB 空間, 但是目前記憶體隻有5MB,那就是記憶體溢出。

javadoc 對 OutOfMemoryError 的解釋是, 沒有空閑記憶體,垃圾收集器也無未能提供更多的記憶體空間 。

java jvm 記憶體溢出和記憶體洩漏的差別

記憶體洩露 memory leak

是指程式運作結束後,沒有釋放已所占用的記憶體空間,一次記憶體洩漏 似乎不會有大的影響,但記憶體洩漏 堆積後的後果就是 記憶體溢出。

java jvm 記憶體溢出和記憶體洩漏的差別

1)單例模式

單例的生命周期和應用程式是一樣長的,是以單例程式中如果持有對外部對象的引用的話,那麼這個外部對象是不能被回收的,則會導緻記憶體洩露的産生。

2)一些提供close的資源未閉導緻記憶體洩漏

資料庫連接配接(dataSource.getConnection() ),網絡連接配接(socket)和 IO流的連接配接必須在finally中 close,否則不能被回收的。

2、二者的關系:

  • 記憶體溢出就是你要的記憶體空間超過了系統實際配置設定給你的空間,此時系統相當于沒法滿足你的需求,就會報記憶體溢出的錯誤。
  • 記憶體溢出:一個盤子用盡各種方法隻能裝4個果子,你裝了5個,結果掉倒地上不能吃了。這就是溢出。比方說棧,棧滿時再做進棧必定産生空間溢出,叫上溢,棧空時再做退棧也産生空間溢出,稱為下溢。就是配置設定的記憶體不足以放下資料項序列,稱為記憶體溢出。說白了就是我承受不了那麼多,那我就報錯。
  • 記憶體洩漏是指你向系統申請配置設定記憶體進行使用(new),可是使用完了以後卻不歸還(delete),結果你申請到的那塊記憶體你自己也不能再通路(也許你把它的位址給弄丢了),而系統也不能再次将它配置設定給需要的程式。就相當于你租了個帶鑰匙的櫃子,你存完東西之後把櫃子鎖上之後,把鑰匙丢了或者沒有将鑰匙還回去,那麼結果就是這個櫃子将無法供給任何人使用,也無法被垃圾回收器回收,因為找不到他的任何資訊。

    記憶體洩漏的堆積最終會導緻記憶體溢出。

3、記憶體溢出的原因以及解決方法

3.1、記憶體溢出的原因

引起記憶體溢出的原因有很多種,小編列舉一下常見的有以下幾種:

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

3.2、記憶體溢出的解決方案:

第一步,修改JVM啟動參數,直接增加記憶體。(-Xms、-Xmx 參數一定不要忘記加)

第二步,檢查錯誤日志,檢視 “OutOfMemory” 錯誤前是否有其它異常或錯誤。

第三步,對代碼進行走查和分析,找出可能發生記憶體溢出的位置。

重點排查以下幾點:

1.檢查對資料庫查詢中,是否有一次獲得全部資料的查詢。
 一般來說,如果一次取十萬條記錄到記憶體,就可能引起記憶體溢出。
 這個問題比較隐蔽,在上線前,資料庫中資料較少,不容易出問題,上線後,
 資料庫中資料多了,一次查詢就有可能引起記憶體溢出。是以對于資料庫查詢盡量采用分頁的方式查詢。
 
2.檢查代碼中是否有死循環或遞歸調用。

3.檢查是否有大循環重複産生新對象實體。

4.檢查對資料庫查詢中,是否有一次獲得全部資料的查詢。
 一般來說,如果一次取十萬條記錄到記憶體,就可能引起記憶體溢出。
 這個問題比較隐蔽,在上線前,資料庫中資料較少,不容易出問題,
 上線後,資料庫中資料多了,一次查詢就有可能引起記憶體溢出。是以對于資料庫查詢盡量采用分頁的方式查詢。

5.檢查List、MAP等集合對象是否有使用完後,未清除的問題。
 List、MAP等集合對象會始終存有對對象的引用,使得這些對象不能被GC回收。