欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

记一次tomcat内存大涨到溢出的经历

程序员文章站 2022-03-21 18:13:01
前一段时间提交了一个产品版本给测试人员测试,测试结果简直出人意料! 测试一段时间后页面就卡死了,当时根据这个现象下意识的怀疑是卡到数据库这一层,然后查看数据库连接相关的参数,如意料之中的相似,连接数太多了!当把数据库连接数解决后,本以为这个bug解决了,但是... 测试一段时间后页面又卡死了!!! ......

前一段时间提交了一个产品版本给测试人员测试,测试结果简直出人意料!

测试一段时间后页面就卡死了,当时根据这个现象下意识的怀疑是卡到数据库这一层,然后查看数据库连接相关的参数,如意料之中的相似,连接数太多了!当把数据库连接数解决后,本以为这个bug解决了,但是...

测试一段时间后页面又卡死了!!!

打开任务管理器,发现tomcat内存超过了1.5g,而且tomcat关不掉!是什么原因导致的呢?左思右想之后,想到了一个可能会导致tomcat内存上涨的点,那就是多线程,然后翻代码找线程池的配置,发现也没什么可疑之处。

那就先解决下tomcat关不掉的问题吧,百度...检查代码...几十分钟后找到了,在tomcat监听器的销毁方法(contextdestroyed)里没有关闭线程池,这种情况下,由于线程池没法关闭,进而导致tomcat无法关闭的问题。

将代码改为:

public class initlistener implements servletcontextlistener{
    private logger logger = logger.getlogger(initlistener.class);
    
    @override
    public void contextinitialized(servletcontextevent sce) {
        logger.info("启动tomcat");
    }

    @override
    public void contextdestroyed(servletcontextevent sce) {
        logger.info("关闭tomcat,关闭线程池");
        classpathxmlapplicationcontext classpathxmlapplicationcontext = new classpathxmlapplicationcontext("classpath*:applicationcontext.xml");
        threadpooltaskexecutor mytaskexecutor = (threadpooltaskexecutor) classpathxmlapplicationcontext.getbean("mytaskexecutor");
        mytaskexecutor.shutdown();
    }

}

好了,tomcat关不掉的问题是解决了。

接下来解决内存溢出的问题(先看日志):

查看tomcat的日志发现,页面每一次调用后台接口spring的配置文件都会初始化一遍,也就是每次请求spring都会重新注入一次bean,而且占用的内存不会被回收!

然后我就想什么情况下会初始化spring的配置文件:tomcat启动的时候;通过关键字new出来的时候,即

 classpathxmlapplicationcontext classpathxmlapplicationcontext = new classpathxmlapplicationcontext("classpath*:applicationcontext.xml");

然后就全局搜索代码找,果然在过滤器里找到了,每次接口来就会new一个对象,多可怕的代码,在心里一直骂自己当时怎么想的!这次经历我会引以为戒,记下来也是告诉自己以后不要再犯类似的问题。

 

欢迎关注微信公众号【java典籍】,收看更多java技术干货!

   ▼微信扫一扫下图↓↓↓二维码关注

 记一次tomcat内存大涨到溢出的经历