導致OutOfMemoryError異常的常見原因有以下幾種:
- 內存中加載的數據量過於龐大,如一次從數據庫取出過多數據;
- 集合類中有對對象的引用,使用完后未清空,使得JVM不能回收;
- 代碼中存在死循環或循環產生過多重復的對象實體;
- 使用的第三方軟件中的BUG;
- 啟動參數內存值設定的過小;
此錯誤常見的錯誤提示:
- tomcat:java.lang.OutOfMemoryError: PermGen space
- weblogic:Root cause of ServletException java.lang.OutOfMemoryError
- tomcat:java.lang.OutOfMemoryError: Java heap space
- resin:java.lang.OutOfMemoryError
- java:java.lang.OutOfMemoryError
解決java.lang.OutOfMemoryError的方法有如下幾種:
(1)增加jvm的內存大小。方法有:
- 在執行某個class文件時候,可以使用java -Xmx256M aa.class來設置運行aa.class時jvm所允許占用的最大內存為256M。
- 對tomcat容器,可以在啟動時對jvm設置內存限度。對tomcat,可以在catalina.bat中添加:
set CATALINA_OPTS=-Xms128M -Xmx256M
set JAVA_OPTS=-Xms128M -Xmx256M
(2)優化程序,釋放垃圾。
主要包括避免死循環,應該及時釋放種資源:內存, 數據庫的各種連接,防止一次載入太多的數據。
Java代碼導致OutOfMemoryError錯誤的解決:
需要重點排查以下幾點:
- 檢查代碼中是否有死循環或遞歸調用。
- 檢查是否有大循環重復產生新對象實體。
- 檢查對數據庫查詢中,是否有一次獲得全部數據的查詢。一般來說,如果一次取十萬條記錄到內存,就可能引起內存溢出。這個問題比較隱蔽,在上線前,數據庫中數據較少,不容易出問題,上線后,數據庫中數據多了,一次查詢就有可能引起內存溢出。因此對於數據庫查詢盡量采用分頁的方式查詢。
- 檢查List、MAP等集合對象是否有使用完后,未清除的問題。List、MAP等集合對象會始終存有對對象的引用,使得這些對象不能被GC回收。
參考Java 內存溢出(java.lang.OutOfMemoryError)的常見情況和處理方式總結