致使OutOfMemoryError異常的常見緣由有如下幾種:java
內存中加載的數據量過於龐大,如一次從數據庫取出過多數據;android
集合類中有對對象的引用,使用完後未清空,使得JVM不能回收;web
代碼中存在死循環或循環產生過多重複的對象實體;數據庫
使用的第三方軟件中的BUG;tomcat
啓動參數內存值設定的太小;服務器
此錯誤常見的錯誤提示:jvm
tomcat:java.lang.OutOfMemoryError: PermGen space優化
tomcat:java.lang.OutOfMemoryError: Java heap spacespa
weblogic:Root cause of ServletException java.lang.OutOfMemoryError.net
resin:java.lang.OutOfMemoryError
java:java.lang.OutOfMemoryError
解決java.lang.OutOfMemoryError的方法有以下幾種:
1、增長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"爲最大內存。
2、 優化程序,釋放垃圾。
主要包括避免死循環,應該及時釋放種資源:內存, 數據庫的各類鏈接,防止一次載入太多的數據。致使java.lang.OutOfMemoryError的根本緣由是程序不健壯。所以,從根本上解決Java內存溢出的惟一方法就是修改程序,及時地釋放沒用的對象,釋放內存空間。 遇到該錯誤的時候要仔細檢查程序,嘿嘿,遇多一次這種問題以後,之後寫程序就會當心多了。
須要重點排查如下幾點:
檢查代碼中是否有死循環或遞歸調用。
檢查是否有大循環重複產生新對象實體。
檢查對數據庫查詢中,是否有一次得到所有數據的查詢。通常來講,若是一次取十萬條記錄到內存,就可能引發內存溢出。這個問題比較隱蔽,在上線前,數據庫中數據較少,不容易出問題,上線後,數據庫中數據多了,一次查詢就有可能引發內存溢出。所以對於數據庫查詢儘可能採用分頁的方式查詢。
檢查List、MAP等集合對象是否有使用完後,未清除的問題。List、MAP等集合對象會始終存有對對象的引用,使得這些對象不能被GC回收。
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 文檔重複佔用內存的目的。
錯誤提示: 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
產生內存溢出的緣由:
出現這個錯誤,通常是由於JVM物理內存太小。默認的Java虛擬機最大內存僅爲64兆,這在開發調試過程當中可能沒有問題,但在實際的應用環境中是遠遠不能知足須要的,除非你的應用很是小,也沒什麼訪問量。不然你可能會發現程序運行一段時間後包java.lang.OutOfMemoryError的錯誤。所以咱們須要提高resin可用的虛擬機內存的大小。
解決方法:
修改/usr/local/resin/bin/httpd.sh中的args選項 添加參數-Xms
(初始內存)和-Xmx
(最大可以使用內存大小)能夠用來限制JVM的物理內存使用量。例如:
args="-Xms128m -Xmx256m"
設置後,JVM初始物理內存是128m,最大能使用物理內存爲256m。
這兩個值應該由系統管理員根據服務器的實際狀況進行設置。
在android開發中也會遇到這樣的問題,通常與圖片加載有關,具體解決方案能夠參考:http://my.oschina.net/u/1244156/blog/222307