`
ayufox
  • 浏览: 272772 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

Tomcat Context reloadabled 与 OutOfMemory(PermSpace)

阅读更多

      我们知道,Sun JVM分代垃圾回收器把堆空间分成3块:

  • Young Gen:年轻代,包括1个Eden区和2个Suvivor区,新创建的对象(大部分为短周期的对象)将进入这个区,虚拟机会频繁地对这个区进行垃圾回收。
  • Old Gen:年老代,当对象在Young Gen呆地足够久(经过几次的垃圾回收仍然存在)或Young Gen空间不足时,对象将进入Old Gen,由于一般是生命周期比较长的对象,因此虚拟机对这块内存的回收频度会比较低,一旦回收,使用的将是一个耗时的Full GC,另外,一旦堆空间不足时,虚拟机也会尝试去回收这个区。
  • Perm Gen:持久代,一些常量定义和类、方法声明及其bytecode都会放在这个区

      问题在于,Perm Gen又是什么样的回收策略呢?

      一直,我都以为Sun JVM对Perm Gen使用的是不垃圾回收的策略,这个观点起源于我之前的一个测试,使用Tomcat Context的reloadabled=”true”,而且项目的jar包比较多,则只要修改几次(事实上是3次)的类定义,然后Tomcat Reload之后,WOW,抛出OutOfMemory(PermSpace)的异常。
      最近,有个同事在做一个项目的时候也碰到这个问题,他是需要应用到IDC环境下的,动态根据配置生成一些新的jar包,在不重新Tomcat的情况下,利用Context的reload属性,自动把新生成的jar加载进来。我当时给他的答案是:

  • 可以使用JVMTI技术,JDK 6新引入的启动后Instrutment,在生成jar包之后,启动Attach到当前JVM,将旧的class卸栽掉(本来也可以使用JVM的debug模式——也是利用这种技术——但debug模式对性能影响会更大)。但限制比较大,类声明不允许发生变更(譬如不能新增属性、方法或者修改方法声明),事实上也不符合他当前的需求。
  • 学过ClassLoader的同学都知道,类定义一旦加载,即使你修改ClassLoader的加载方式,强制再加载一次,也是不行的,因为加载的类定义会被JVM Cache,是不允许重新再加载的。那么Tomcat reloadable是怎么做的呢?因为JVM识别两个类定义是否一致的方法是根据加载的ClassLoader和类全名(即包括package名和类名),那么Tomcat reloadable可以重新加载同一个类定义方法是,将context旧的WebappClassLoader丢弃,创建一个新的WebappClassLoader来达到加载新的类定义的。其实就是将context stop再start,所以代价非常昂贵。这种方式相当于在PermSpace又存放了一份jar定义,所以会导致PermSpace猛增。出现OutOfMemory(PermSpace),说明对旧的class没有回收,建议可以看看JVM的启动参数,有没有对PermSpace垃圾回收的选项。

        由于我之前错误的观点,我以为Sun JVM的PermSpace是不垃圾回收的,为了证明这一点,我查了一些资料;

JVM参考规范 写道
The method area is created on virtual machine start-up. Although the method area is logically part of the heap, simple implementations may choose not to either garbage collect or compact it. This version of the Java virtual machine specification does not mandate the location of the method area or the policies used to manage compiled code. The method area may be of a fixed size or may be expanded as required by the computation and may be contracted if a larger method area becomes unnecessary. The memory for the method area does not need to be contiguous.


       JVM规范并没有规定实现对方法区(PermSpace)一定要垃圾回收,“simple implementations may choose not to either garbage collect or compact it”。很不爽的”may”,当然不能作为问题的答案。
       重新猜想,也许JVM的启动参数是可以指定的,确实如此,然而确是很明确地指出我的想法是错误的:

-Xnoclassgc
Disable class garbage collection. Use of this option will prevent memory recovery from loaded classes thus increasing overall memory usage. This could cause OutOfMemoryError to be thrown in some applications.

 

      也就是说,默认情况下,JVM是会对这块区域进行垃圾回收的。难道是由于Tomcat的策略导致的?我们一步一步地来进行验证
      准备工作:建立测试项目test2(引用的jar包不要太大也不能太小,我建立的项目启动后JVM的PermSpace占用大概20M),在tomcat下配置该应用的reloadabled=”true”。
       猜测1:Tomcat废弃WebappClassLoader没有被置null,导致相应的引用类都没有非卸载。验证步骤如下(Tomcat测试版本是6.0.14):
       步骤1:启动Tomcat,并启动JConsole,监控,并dump下堆内存。
       步骤2:多次修改测试代码,引起Tomcat的reload生效,每次强制在JConsole上进行GC,并dump下堆内存。
       结果如下(JConsole监控图):
 
       可以看到,PermSpace在reload的时候,内存会有一个陡增,第一个陡增没有下降下来,第二个陡增后都可以通过GC下降下来,猜测是Tomcat对一个Context,在reload后会维持两个WebappClassLoader,一个当前的WebappClassLoader和一个历史的WebappClassLoader。我们通过jhat分析dump出来的堆文件证实了有2个WebapClassLoader这种猜测

2 instances of class org.apache.catalina.deploy.NamingResources
2 instances of class org.apache.catalina.loader.WebappClassLoader
2 instances of class org.apache.catalina.mbeans.MBeanFactory

      注意:用其他的tomcat版本做了相同的测试,6.0.20版本与此版本测试结果一致,6.0.26版本结果会比较好,只维持一个WebappClassLoader,即上面的JConsole曲线第一次上升之后也可以通过GC降下来。
      猜测2:Tomcat在新的WebappClassLoader把其他类加载进来之前,不会将旧的WebappClassLoader置null。验证步骤如下(Tomcat测试版本是6.0.14):
      我们先看一下Tomcat中context reload方法,可以看到reload其实就是一次stop和一次start

    public synchronized void reload() {
        // Validate our current component state
        if (!started)
            throw new IllegalStateException
                (sm.getString("containerBase.notStarted", logName()));
        setPaused(true);
        try {
            stop();
        } catch (LifecycleException e) {
            log.error(sm.getString("standardContext.stoppingContext"), e);
        }
        try {
            start();
        } catch (LifecycleException e) {
            log.error(sm.getString("standardContext.startingContext"), e);
        }
        setPaused(false);
    }

 

       步骤1:将tomcat的JVM启动参数中设置调试模式,在StanardContext的reload的start那里设一个断点,确保start之前会停住,我们将观察这种时候进行垃圾回收能否将PermSpace空间回收
       步骤2:启动JConsole,attach到Tomcat进程上
       步骤3:修改测试代码,引起Tomcat的reload生效,在第二次reload  start停止的时候,使用JConsole强行做一个垃圾回收(注意,如果是6.0.26版本,应该在第一次reload的时候观察)
        可以看到JConsole的图如下
 
       可以看到,在start停住的时候,我们进行一个GC后,PermSpace会有一个陡降,说明在stop的时候,已经会把WebappClassLoader中的内存释放掉
       猜测3:JVM在PermSpace空间不足的时候,并不会试图先去回收PermSpace
       一般情况下,我们当然无法在start之前停住,即使在这个阶段强行做一个System.gc(),然而gc的调用无法预料的,假设在start之前PermSpace内存没有被释放,而start的时候又会在PermSpace分配大量的内存,如果此时PermSpace空间不足,是否会引起JVM对PermSpace进行垃圾回收呢?
       验证步骤如下(Tomcat测试版本是6.0.14):
       步骤1:从上面我们可以看到,2个WebappClassLoader及其他classloader加载的类占用的PermSpace空间大概是25M,因此我们在启动参数中设置PermSpace最大的空间值为26M(-XX:MaxPermSize=26MB)
       步骤2:修改测试代码,引起reload,在第2次reload的时候。WOW,OutOfMemory

Caused by: java.lang.OutOfMemoryError: PermGen space
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClass(ClassLoader.java:620)
at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:124)

       可以看到,非常遗憾,JVM在PermSpace空间不足的时候,并不会先主动去释放PermSpace空间。
       我们可以得出如下结果:

  • 对于Tomcat6.0.20及以下版本,要保证reload的时候不出现OutOfMemory的错误,必须维持PermSpace空间为:其他loader加载的类定义占的空间+3 * 应用类占的空间(WebappClassLoader加载)
  • 对于Tomcat6.0.26及以下版本,要保证reload的时候不出现OutOfMemory的错误,必须维持PermSpace空间为:其他loader加载的类定义占的空间+2 * 应用类占的空间(WebappClassLoader加载)

 

4
3
分享到:
评论
5 楼 javaeyes 2011-01-26  
为啥我看到是tomcat和glassfish从来不回收perm space ? 项目里面有数据库连接池,spring 引入的cglib
4 楼 ayufox 2010-04-27  
实际上在tomcat 6.0.20以后的已经可以保证ClassLoader已经不被持有了,见上面的猜想二的论证过程,在tomcat 6.0.20中的问题是,JVM并不保证在PermSpace不足时会去释放PermSpace,这是无论换了什么tomcat版本都没有办法解决的事情,能换的也就只有JVM了
3 楼 melin 2010-04-27  
这个问题,你可以参考一下这个帖子:
Tomcat 7将解决web应用reload导致的OutOfMemoryError(OOME)问题

http://pt.alibaba-inc.com/wp/experience_530/tomcat-7-will-address-web-application-reload-caused-outofmemoryerror-oome-problem-3.html
2 楼 ayufox 2010-04-18  
关于这个问题,我猜想JVM的hotspot机制对bytecode的优化后的code并不会放在PermSpace,在是放在CodeCache这个区(上面监控图中非堆部分的左区域)。 并非我不愿做这个测试,而是这个测试实在太困难了,一方面,无法预料到JVM具体的hotspot策略,另一方面,即使bytecode做了优化,也是极少部分做了优化,很难从监控图中能够看出来
1 楼 dwangel 2010-04-18  
关于这个测试,我还有一点猜想:
Sun JVM 是 hotspot机制,也就是说,会在运行中,根据执行情况优化二进制码。
单纯 start stop,不能反映perm space里这部分数据。
最好是web app start之后,调用应用100次以后关闭。
然后 再stop,看看能不能降下来。

相关推荐

Global site tag (gtag.js) - Google Analytics