适用场景

Tomcat 运行过程遇到Caused by: java.lang.OutOfMemoryError: PermGen space或者java.lang.OutOfMemoryError: Java heap 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)那么就会产生此错误信息了。

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 最大不要超过可用物理内存的80%,一般的要将-Xms和-Xmx选项设置为相同,而-Xmn为1/4的-Xmx值。

解决办法

linux下的tomcat

修改$TOMCAT_HOME/bin/catalina.sh文件:

# OS specific support.  $var must be set to either true or false.

JAVA_OPTS="-server -Xms2048m -Xmx2048m -XX:PermSize=128M -XX:MaxNewSize=256m -XX:MaxPermSize=512m -Djava.awt.headless=true"

加入以上JAVA_OPTS行记录后重启tomcat服务,其中JAVA_OPTS各参数的具体数值大小须根据实际服务器物理内存大小进行分配。

windows下的tomcat

修改tomcat的bin目录下的catalina.bat

set JAVA_OPTS=-Xms2048m -Xmx2048m -XX:PermSize=128M -XX:MaxNewSize=256m -XX:MaxPermSize=512m -Djava.awt.headless=true

加入以上JAVA_OPTS行记录后重启tomcat服务,其中JAVA_OPTS各参数的具体数值大小须根据实际服务器物理内存大小进行分配。

Eclipse修改tomcat内存

Windows → perferences→ Servers → Runtime Environm → 选中你需要的tomcat 点击 Edit → Installed JREs → 双击jdk → 在白框处写入-Xms2048m -Xmx2048m -XX:PermSize=128M -XX:MaxNewSize=256m -XX:MaxPermSize=512m -Djava.awt.headless=true即可。

监控JVM

查看java进程

linux和windows操作一致:

[aaa@localhost conf]$jps -v

15618 Bootstrap -Djava.util.logging.config.file=/home/aaa/tomcat8_9219/conf/logging.properties -Djava.util.logging.manager=org.apache.juli.ClassLoaderLogManager -Djdk.tls.ephemeralDHKeySize=2048 -Djava.protocol.handler.pkgs=org.apache.catalina.webresources -Dcatalina.base=/home/aaa/tomcat8_9219 -Dcatalina.home=/home/aaa/tomcat8_9219 -Djava.io.tmpdir=/home/aaa/tomcat8_9219/temp

可看到Bootstrap 前面的数字10142即为Java进程号,同时可看到JAVA_OPTS参数生效情况。

查看JVM的GC情况

使用jstat工具查看,语法为:

jstat [ generalOption | outputOptions vmid [intervals|ms] ]参数解释:

*Options   -选项,我们一般使用-gcutil查看GC情况
vmid      -VM的进程号,即当前运行的java进程号
interval   -间隔时间,单位为秒或毫秒

本例:查看java进程号15618且每隔3秒输出1次,执行# jstat -gcutil 15618 3000

[aaa@localhost conf]$ jstat  -gcutil  15618 3000
  S0     S1     E      O      M     CCS    YGC     YGCT    FGC    FGCT     GCT   
 65.29   0.00  44.42  13.98  98.00  96.44     14    0.789     3    0.427    1.216
 65.29   0.00  44.42  13.98  98.00  96.44     14    0.789     3    0.427    1.216
 65.29   0.00  44.42  13.98  98.00  96.44     14    0.789     3    0.427    1.216
 65.29   0.00  44.42  13.98  98.00  96.44     14    0.789     3    0.427    1.216
 65.29   0.00  44.42  13.98  98.00  96.44     14    0.789     3    0.427    1.216
 65.29   0.00  44.42  13.98  98.00  96.44     14    0.789     3    0.427    1.216
  • 参数说明如下:
S0     -Heap上的Survivor space 0区已使用空间的百分比
S1     -Heap上的Survivor space 1区已使用空间的百分比
E      -Heap上Eden space区已使用空间的百分比
O      -Heap上的Old space区已使用空间的百分比,达到100%,就是内存溢出了
P       -Perm space区已使用空间的百分比,达到100%,就是内存溢出了
YGC    -从应用程序启动到采样时发生Yang GC 的次数
YGCT   -从应用程序启动到采样时Yang GC所用的时间【单位秒】
FGC    -从应用程序启动到采样时Full GC的次数
FGCT   -从应用程序启动到采样时Full GC所用的时间
GCT    -从应用程序启动到采样时用于垃圾回收的总时间【单位秒】

通过监控O、P数值情况,再次调整JAVA_OPTS相关参数大小达到优化目的。