天天看點

【高并發】高并發環境下如何防止Tomcat記憶體溢出?看完我懂了!!

随着系統并發量越來越高,Tomcat所占用的記憶體就會越來越大,如果對Tomcat的記憶體管理不當,則可能會引發Tomcat記憶體溢出的問題,那麼,如何防止Tomcat記憶體溢出呢?我們今天就來一起探讨下這個問題。

防止Tomcat記憶體溢出可以總結為兩個方案:一個是設定Tomcat啟動的初始記憶體,一個是防止Tomcat所用的JVM記憶體溢出。接下來,我們就分别對這兩種方案作出簡單的介紹。

其初始空間(即-Xms)是實體記憶體的1/64,最大空間(-Xmx)是實體記憶體的1/4。可以利用JVM提供的-Xmn -Xms -Xmx等選項可進行設定。

執行個體

以下給出1G記憶體環境下java jvm 的參數設定參考:

Linux

在/usr/local/apache-tomcat-7.0/bin 目錄下的catalina.sh檔案中,添加:JAVA_OPTS=’-Xms512m -Xmx1024m’,要加“m”說明是MB,否則就是KB了,在啟動tomcat時會報記憶體不足。

-Xms:初始值

-Xmx:最大值

-Xmn:最小值

Windows

在catalina.bat最前面加入set JAVA_OPTS=-Xms128m -Xmx350m,如果用startup.bat啟動tomcat,OK設定生效。夠成功的配置設定200M記憶體。但是如果不是執行startup.bat啟動tomcat而是利用windows的系統服務啟動tomcat服務,上面的設定就不生效了,就是說set JAVA_OPTS=-Xms128m -Xmx350m 沒起作用。上面配置設定200M記憶體就OOM了。。

windows服務執行的是bin\tomcat.exe。它讀取系統資料庫中的值,而不是catalina.bat的設定。

解決辦法

修改系統資料庫

原值為

加入 -Xms300m -Xmx350m

重起tomcat服務,設定生效。

解釋

Heap size 設定

JVM堆的設定是指java程式運作過程中JVM可以調配使用的記憶體空間的設定.JVM在啟動的時候會自動設定Heap size的值,其初始空間(即-Xms)是實體記憶體的1/64,最大空間(-Xmx)是實體記憶體的1/4。可以利用JVM提供的-Xmn -Xms -Xmx等選項可進行設定。Heap size 的大小是Young Generation 和Tenured Generaion 之和。

提示:在JVM中如果98%的時間是用于GC且可用的Heap size 不足2%的時候将抛出此異常資訊。

提示:Heap Size 最大不要超過可用實體記憶體的80%,一般的要将-Xms和-Xmx選項設定為相同,而-Xmn為1/4的-Xmx值。

解決方法

手動設定Heap size

修改TOMCAT_HOME/bin/catalina.bat,在“echo “Using CATALINA_BASE: $CATALINA_BASE””上面加入以下代碼。

或修改catalina.sh

在“echo “Using CATALINA_BASE: $CATALINA_BASE””上面加入以下行:

原因

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.bat(Linux下為catalina.sh),在代碼

“echo “Using CATALINA_BASE: $CATALINA_BASE””上面加入以下行:

catalina.sh檔案的修改如下。

Java代碼

發現很多人把問題歸因于: spring,hibernate,tomcat,因為他們動态産生類,導緻JVM中的permanent heap溢出 。然後解決方法衆說紛纭,有人說更新 tomcat版本到最新甚至幹脆不用tomcat。還有人懷疑spring的問題,在spring論壇上讨論很激烈,因為spring在AOP時使用CBLIB會動态産生很多類。

但問題是為什麼這些王牌的開源會出現同一個問題呢,那麼是不是更基礎的原因呢?tomcat在Q&A很隐晦的回答了這一點,我們知道這個問題,但這個問題是由一個更基礎的問題産生。

于是有人對更基礎的JVM做了檢查,發現了問題的關鍵。原來SUN 的JVM把記憶體分了不同的區,其中一個就是permenter區用來存放用得非常多的類和類描述。本來SUN設計的時候認為這個區域在JVM啟動的時候就固定了,但他沒有想到現在動态會用得這麼廣泛。而且這個區域有特殊的垃圾收回機制,現在的問題是動态加載類到這個區域後,gc根本沒辦法回收!

對于以上兩個問題,我的處理是:

在catalina.bat的第一行增加:

在catalina.sh的第一行增加:

如果覺得文章對你有點幫助,請微信搜尋并關注「 冰河技術 」微信公衆号,跟冰河學習高并發程式設計技術。

最後,附上并發程式設計需要掌握的核心技能知識圖,祝大家在學習并發程式設計時,少走彎路。

【高并發】高并發環境下如何防止Tomcat記憶體溢出?看完我懂了!!