1. 程式人生 > >ElasticSearch——Java 記憶體溢位

ElasticSearch——Java 記憶體溢位

Java 記憶體溢位(java.lang.OutOfMemoryError)的常見情況和處理方式總結

java.lang.OutOfMemoryError這個錯誤我相信大部分開發人員都有遇到過,產生該錯誤的原因大都出於以下原因:JVM記憶體過小、程式不嚴密,產生了過多的垃圾。

導致OutOfMemoryError異常的常見原因有以下幾種:

  1. 記憶體中載入的資料量過於龐大,如一次從資料庫取出過多資料;
  2. 集合類中有對物件的引用,使用完後未清空,使得JVM不能回收;
  3. 程式碼中存在死迴圈或迴圈產生過多重複的物件實體;
  4. 使用的第三方軟體中的BUG;
  5. 啟動引數記憶體值設定的過小;

此錯誤常見的錯誤提示:

  1. tomcat:java.lang.OutOfMemoryError: PermGen space
  2. tomcat:java.lang.OutOfMemoryError: Java heap space
  3. weblogic:Root cause of ServletException java.lang.OutOfMemoryError
  4. resin:java.lang.OutOfMemoryError
  5. java:java.lang.OutOfMemoryError

解決java.lang.OutOfMemoryError的方法有如下幾種:

一、增加jvm的記憶體大小。方法有: 1)在執行某個class檔案時候,可以使用java -Xmx256M aa.class來設定執行aa.class時jvm所允許佔用的最大記憶體為256M。 2)對tomcat容器,可以在啟動時對jvm設定記憶體限度。對tomcat,可以在catalina.bat中新增:

set CATALINA_OPTS=-Xms128M -Xmx256M
set JAVA_OPTS=-Xms128M -Xmx256M

或者把%CATALINA_OPTS%和%JAVA_OPTS%代替為-Xms128M -Xmx256M

3)對resin容器,同樣可以在啟動時對jvm設定記憶體限度。在bin資料夾下建立一個startup.bat檔案,內容如下:

@echo off
call "httpd.exe"  "-Xms128M" "-Xmx256M"
:end

其中"-Xms128M"為最小記憶體,"-Xmx256M"為最大記憶體。

二、 優化程式,釋放垃圾。

主要包括避免死迴圈,應該及時釋放種資源:記憶體, 資料庫的各種連線,防止一次載入太多的資料。導致java.lang.OutOfMemoryError的根本原因是程式不健壯。因此,從根本上解決Java記憶體溢位的唯一方法就是修改程式,及時地釋放沒用的物件,釋放記憶體空間。 遇到該錯誤的時候要仔細檢查程式,嘿嘿,遇多一次這種問題之後,以後寫程式就會小心多了。

Java程式碼導致OutOfMemoryError錯誤的解決:

需要重點排查以下幾點:

  1. 檢查程式碼中是否有死迴圈或遞迴呼叫。
  2. 檢查是否有大迴圈重複產生新物件實體。
  3. 檢查對資料庫查詢中,是否有一次獲得全部資料的查詢。一般來說,如果一次取十萬條記錄到記憶體,就可能引起記憶體溢位。這個問題比較隱蔽,在上線前,資料庫中資料較少,不容易出問題,上線後,資料庫中資料多了,一次查詢就有可能引起記憶體溢位。因此對於資料庫查詢儘量採用分頁的方式查詢。
  4. 檢查List、MAP等集合物件是否有使用完後,未清除的問題。List、MAP等集合物件會始終存有對物件的引用,使得這些物件不能被GC回收。