1. 程式人生 > >Java 記憶體溢位(java.lang.OutOfMemoryError: Java heap space)分析與解決

Java 記憶體溢位(java.lang.OutOfMemoryError: Java heap space)分析與解決

說明:下面出現的問題為本人在myeclips開發過程中,開發工具時獲取海量資料時出現的問題報錯。由於本人開發電腦使用的4g記憶體,虛擬記憶體與myeclips相關記憶體設定無法滿足要求。

問題分析:(網上資料整合與翻譯)

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回收。

tomcat中java.lang.OutOfMemoryError: PermGen space異常處理

PermGen space的全稱是Permanent Generation space,是指記憶體的永久儲存區域,這塊記憶體主要是被JVM存放Class和Meta資訊的,Class在被Loader時就會被放到PermGen space中, 它和存放類例項(Instance)的Heap區域不同,GC(Garbage Collection)不會在主程式執行期對PermGen space進行清理,所以如果你的應用中有很多CLASS的話,就很可能出現PermGen space錯誤, 這種錯誤常見在web伺服器對JSP進行pre compile的時候。如果你的WEB APP下都用了大量的第三方jar, 其大小超過了jvm預設的大小(4M)那麼就會產生此錯誤資訊了。 解決方法: 手動設定MaxPermSize大小修改TOMCAT_HOME/bin/catalina.sh在

echo "Using CATALINA_BASE:   $CATALINA_BASE"

上面加入以下行:

JAVA_OPTS="-server -XX:PermSize=64M -XX:MaxPermSize=128m

建議:將相同的第三方jar檔案移置到tomcat/shared/lib目錄下,這樣可以達到減少jar 文件重複佔用記憶體的目的。

weblogic中java.lang.OutOfMemoryError異常處理

錯誤提示: Root cause of ervletException java.lang.OutOfMemoryError 解決辦法:調整bea/weblogic/common中CommEnv中引數

    :sun
  if "%PRODUCTION_MODE%" == "true" goto sun_prod_mode
  set JAVA_VM=-client
  set MEM_ARGS=-Xms256m -Xmx512m -XX:MaxPermSize=256m
  set JAVA_OPTIONS=%JAVA_OPTIONS% -Xverify:none
  goto continue
  :sun_prod_mode
  set JAVA_VM=-server
  set MEM_ARGS=-Xms256m -Xmx512m -XX:MaxPermSize=256m
  goto continue

Resin下java.lang.OutOfMemoryError異常處理

產生記憶體溢位的原因:

出現這個錯誤,一般是因為JVM實體記憶體過小。預設的Java虛擬機器最大記憶體僅為64兆,這在開發除錯過程中可能沒有問題,但在實際的應用環境中是遠遠不能滿足需要的,除非你的應用非常小,也沒什麼訪問量。否則你可能會發現程式執行一段時間後包java.lang.OutOfMemoryError的錯誤。因此我們需要提升resin可用的虛擬機器記憶體的大小。

解決方法:

修改/usr/local/resin/bin/httpd.sh中的args選項 新增引數-Xms(初始記憶體)和-Xmx(最大能夠使用記憶體大小)可以用來限制JVM的實體記憶體使用量。例如:

args="-Xms128m -Xmx256m"

設定後,JVM初始實體記憶體是128m,最大能使用實體記憶體為256m。

這兩個值應該由系統管理員根據伺服器的實際情況進行設定。

**個人問題解決方法:** 本人是開發海量資料分析的小工具,在myeclips執行環境中報錯,但是直接匯出為jar之後執行就不會出現問題。在此之後匯出為exe檔案繼續執行也沒有發現問題。