读书人

Tomcat的JreMemoryLeakPreventionList

发布时间: 2012-09-14 11:53:44 作者: rapoo

Tomcat的JreMemoryLeakPreventionListener1小时(hourly)执行一次full gc

项目部署在Tomcat7上,操作系统是centOS5.5,打印出gc的日志,发现有些full gc在heap还很空余的时候执行,后来仔细一看,发现一小时执行一次,日志如下:

2011-12-13T19:01:36.706+0800: 1.111: [Full GC (System) [PSYoungGen: 1936K->0K(597312K)] [PSOldGen: 0K->1860K(1365376K)] 1936K->1860K(1962688K) [PSPermGen: 10426K->10426K(21248K)], 0.0461030 secs] [Times: user=0.06 sys=0.00, real=0.04 secs]?

2011-12-13T20:01:36.816+0800: 3601.221: [Full GC (System) [PSYoungGen: 33170K->0K(597312K)] [PSOldGen: 1860K->30610K(1365376K)] 35030K->30610K(1962688K) [PSPermGen: 52266K->52266K(52352K)], 0.3710140 secs] [Times: user=0.37 sys=0.01, real=0.37 secs]?

2011-12-13T21:01:37.201+0800: 7201.606: [Full GC (System) [PSYoungGen: 3424K->0K(597312K)] [PSOldGen: 30610K->33665K(1365376K)] 34034K->33665K(1962688K) [PSPermGen: 52400K->52400K(83968K)], 0.3356130 secs] [Times: user=0.34 sys=0.00, real=0.33 secs]?

?

然后google,发现是Tomcat的JreMemoryLeakPreventionListener调用的full gc,目的是为了方式内存泄露,其中的代码如下:

?

?

               synchronized(GC.lock)                {                    l = GC.latencyTarget;                    if(l != 9223372036854775807L)                        break label0;                    GC.daemon = null;                }                return;            }            long l1 = GC.maxObjectInspectionAge();            if(l1 >= l)            {                System.gc();                l1 = 0L;            }       

可以看到调用了System.gc(),有以下几种办法可以停止这个一小时一次的full gc:
1.通过添加jvm参数?-XX:+DisableExplicitGC? 来制止显式调用的full gc,这里需要说一点:有些帖子说-XX:-DisableExplicitGC是停止full gc,-XX:+DisableExplicitGC是启动full gc,纯属扯淡。
2.给JreMemoryLeakPreventionListener添加一个参数,如下
<Listener className="org.apache.catalina.core.JreMemoryLeakPreventionListener"gcDaemonProtection="false"/>
这样的话还是会有守护进程,只是不执行full gc了。
3.把这个listener从tomcat的server.xml里去掉。

我采用的是第2中方法。







?

读书人网 >Web前端

热点推荐