一、内存溢出的解决
1、java.lang.OutOfMemoryError: PermGen space
最近TOMCAT老是自动挂掉报错的日志是
25-Feb-2021 10:12:33.568 SEVERE [Thread-31] org.apache.coyote.AbstractProtocol$ConnectionHandler.process Failed to complete processing of a request
java.lang.OutOfMemoryError: PermGen space
翻译下来是:PermGen存储空间不够,导致内存溢出
PermGen是什么? PermGen space的全称是Permanent Generation space,内存的永久保存区域
内存益出:这一部分用于存放Class和Meta的信息,Class在被Load的时候被放入PermGen space区域,它和和存放Instance的Heap区域不同,GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的APP会LOAD很多CLASS的话,就很可能出现PermGen space错误。
执行jstat命令查看对应进程内存分配情况,TOMCAT的进程是ID53749
jstat -gcpermcapacity 53749
通过查询得到结果:Perm初始的只有21MB,PGCMAX(Perm代的最大容量)的值为82MB,值太小了,所以内存很容易溢出
通过修改catalina.sh增加Perm最大容量的设置catalina.sh文件加入
JAVA_OPTS="-XX:+UseParallelGC -XX:+UseParallelOldGC -Xms2048m -Xmx4096M -Xss1024K -XX:PermSize=512m -XX:MaxPermSize=8192M -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintGCDateStamps -XX:+PrintHeapAtGC -Xloggc:../logs/gc.log"
增加配置完重启tomcat,就可以看,PGCMN初始Perm的值为512MB,到PGCMAX的值已经改为了8GB大小比之前的内存就大了很多。
PGCMN | Perm代中初始化(最小)的大小 (KB) |
PGCMX | Perm代的最大容量 (KB) |
PGC | Perm代当前新生成的容量 (KB) |
PC | Perm(持久代)的容量 (KB) |
YGC | 从应用程序启动到采样时年轻代中gc次数 |
FGC | 从应用程序启动到采样时old代(全gc)gc次数 |
FGCT | 从应用程序启动到采样时old代(全gc)gc所用时间(s) |
GCT | 从应用程序启动到采样时gc用的总时间(s) |
查看堆内存使用情况 jmap -heap 进程号
通过Arthas也可以更进一步看清楚JVM整个内存的分配情况。