Java 記憶體溢位(java.lang.OutOfMemoryError: Java heap space)分析與解決
說明:下面出現的問題為本人在myeclips開發過程中,開發工具時獲取海量資料時出現的問題報錯。由於本人開發電腦使用的4g記憶體,虛擬記憶體與myeclips相關記憶體設定無法滿足要求。
問題分析:(網上資料整合與翻譯)
java.lang.OutOfMemoryError這個錯誤我相信大部分開發人員都有遇到過,產生該錯誤的原因大都出於以下原因:JVM記憶體過小、程式不嚴密,產生了過多的垃圾。
導致OutOfMemoryError異常的常見原因有以下幾種:
- 記憶體中載入的資料量過於龐大,如一次從資料庫取出過多資料;
- 集合類中有對物件的引用,使用完後未清空,使得JVM不能回收;
- 程式碼中存在死迴圈或迴圈產生過多重複的物件實體;
- 使用的第三方軟體中的BUG;
- 啟動引數記憶體值設定的過小;
此錯誤常見的錯誤提示:
- tomcat:java.lang.OutOfMemoryError: PermGen space
- tomcat:java.lang.OutOfMemoryError: Java heap space
- weblogic:Root cause of ServletException java.lang.OutOfMemoryError
- resin:java.lang.OutOfMemoryError
- 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錯誤的解決:
需要重點排查以下幾點:
- 檢查程式碼中是否有死迴圈或遞迴呼叫。
- 檢查是否有大迴圈重複產生新物件實體。
- 檢查對資料庫查詢中,是否有一次獲得全部資料的查詢。一般來說,如果一次取十萬條記錄到記憶體,就可能引起記憶體溢位。這個問題比較隱蔽,在上線前,資料庫中資料較少,不容易出問題,上線後,資料庫中資料多了,一次查詢就有可能引起記憶體溢位。因此對於資料庫查詢儘量採用分頁的方式查詢。
- 檢查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檔案繼續執行也沒有發現問題。