天天看點

Tomcat記憶體溢出及大小調整

一、

tomcat記憶體設定問題 收藏

    在使用Java程式從資料庫中查詢大量的資料或是應用伺服器(如tomcat、jboss,weblogic)加載jar包時會出現java.lang.OutOfMemoryError異常。這主要是由于應用伺服器的記憶體不足引起的。這種異常常有以下幾種情況(以下以tomcat環境為例,其它WEB伺服器如jboss,weblogic等是同一個道理):

    1.  java.lang.OutOfMemoryError: PermGen space

        PermGen space的全稱是Permanent Generation space,是指記憶體的永久儲存區域OutOfMemoryError: PermGen space。從文字上看就是記憶體溢出,解決方法是加大記憶體。為什麼會記憶體溢出,這是由于這塊記憶體主要是被JVM存放Class和Meta資訊的,Class在被Load的時候被放入PermGen space區域,它和存放Instance的Heap區域不同,GC(Garbage Collection)不會在主程式運作期對PermGen space進行清理,是以如果你的APP會LOAD很多CLASS的話,就很可能出現PermGen space錯誤。這種錯誤常見在web伺服器對JSP進行pre compile的時候。如果你的WEB APP下都用了大量的第三方jar, 其大小超過了jvm預設的大小(4M)那麼就會産生此錯誤資訊了。

        解決方法: 手動設定MaxPermSize大小

        a.如果tomcat是以bat方式啟動的,則如下設定:

        修改TOMCAT_HOME/bin/catalina.sh

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

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

        b.如果tomcat是注冊成了windows服務,以services方式啟動的,則需要修改系統資料庫中的相應鍵值。

            打開系統資料庫,找到目錄HKEY_LOCAL_MACHINE\SOFTWARE\Apache Software Foundation\Procrun 2.0\htfty\Parameters\Java,其中目錄位址中紅色标注的(如htfty)需要根據不同情況作修改,為tomcat服務注冊成windows服務的名稱。 可以看到JvmMs和JvmMx項,其中JvmMs設定最小的記憶體使用參數,JvmMx設定最大的記憶體使用參數。設定好JvmMs和JvmMx項的值,重新開機tomcat伺服器即可生效。

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

     2.  java.lang.OutOfMemoryError: Java heap space

           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

    a.如果tomcat是以bat方式啟動的,則如下設定:

    修改TOMCAT_HOME/bin/catalina.sh

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

    JAVA_OPTS="-server -Xms800m -Xmx800m    -XX:MaxNewSize=256m"

    b.如果tomcat是注冊成了windows服務,以services方式啟動的,則需要修改系統資料庫中的相應鍵值。

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

二、Tomcat本身不能直接在計算機上運作,需要依賴于硬體基礎之上的作業系統和一個java虛拟機。JAVA程式啟動時JVM都會配置設定一個初始記憶體和最大記憶體給這個應用程式。這個初始記憶體和最大記憶體在一定程度都會影響程式的性能。比如說在應用程式用到最大記憶體的時候,JVM是要先去做垃圾回收的動作,釋放被占用的一些記憶體。是以想調整Tomcat的啟動時初始記憶體和最大記憶體就需要向JVM聲明,一般的JAVA程式在運作都可以通過中-Xms -Xmx來調整應用程式的初始記憶體和最大記憶體: 這兩個值的大小一般根據需要進行設定。初始化堆的大小執行了虛拟機在啟動時向系統申請的記憶體的大小。一般而言,這個參數不重要。但是有的應用程式在大負載的情況下會急劇地占用更多的記憶體,此時這個參數就是顯得非常重要,如果虛拟機啟動時設定使用的記憶體比較小而在這種情況下有許多對象進行初始化,虛拟機就必須重複地增加記憶體來滿足使用。由于這種原因,我們一般把-Xms和-Xmx設為一樣大,而堆的最大值受限于系統使用的實體記憶體。一般使用資料量較大的應用程式會使用持久對象,記憶體使用有可能迅速地增長。當應用程式需要的記憶體超出堆的最大值時虛拟機就會提示記憶體溢出,并且導緻應用服務崩潰。是以一般建議堆的最大值設定為可用記憶體的最大值的80%。

     Tomcat預設可以使用的記憶體為128MB,在較大型的應用項目中,這點記憶體是不夠的,需要調大。有以下幾種方法可以選用:

第一種方法:

Windows下,在檔案/bin/catalina.bat,Unix下,在檔案/bin/catalina.sh的前面,增加如下設定:

JAVA_OPTS='-Xms【初始化記憶體大小】 -Xmx【可以使用的最大記憶體】'

需要把這個兩個參數值調大。例如:

JAVA_OPTS='-Xms256m -Xmx512m'

表示初始化記憶體為256MB,可以使用的最大記憶體為512MB。

第二種方法: 環境變量中設     變量名:JAVA_OPTS     變量值:-Xms512m   -Xmx512m