问题场景
公司linux服务器老是内存不够用,但是实际部署在上面的java项目只有两个而已;
问题逐步解决的过程
1.刚开始的时候发现有几次used的使用量每隔一段时间就增加一点,而停用tomat的时候就会减少很多,再加上服务器上的日志爆出了 堆栈溢出的异常,通过jdk中的工具的分析,发现了是执行查询计划语句的问题(因为我们项目是ssh项目,而我们在项目中大多使用了hql以及原生sql语句来做查询,但是由于经验缺少的原因,除了hql是使用占位符来填入可变变量的,原生sql是直接通过字符串拼接的,结果这种无心的操作居然造成了堆栈溢出,因为这些可变的变量被拼装起来的语句,就会使得这一个语句结果变成不同的字符串变量,而hibernate本身会缓存执行计划,所以就会因为存储的变量过多而导致内存消耗,堆栈溢出);解决完这个问题后,项目似乎并没有像这样子频繁的出现内存不够用的情况了
一段时间的风平浪静后,随之而来的狂狼总是让人措手不及~~
2.used又狂飙了,这一次通过top命令,我发现了奇怪的现象,我发现里面运行的java项目居然不只是2个而已,居然有8个,8个呀,我真的是一脸懵逼~~~~,后来为了确认是不是我看错了,我使用了命令 ps -ef|grep java | grep catalina 来查看我们的tomcat下面到底运行了多少个java进程,结果不得了,居然真的有8个,当时的下巴直往下掉,就像汤姆猫吃惊的表情一样T_T, 为此我脑中飘过一个疑问,我每次发布项目都是先关闭tomcat再上传项目war包,再启动tomcat的,照理说之前的java进程应该会随着tomcat的关闭同时关闭掉的,为什么会成为一个僵尸进程呢? 怀抱疑问,我投入了度娘的怀抱。通过网络博文,以及结合自身项目在这段期间的修改找到了问题的原因,而这个原因居然和定时任务有关,原因是定时任务其实就是一个用户线程,而我们代码中使用了多个定时任务,它们并不会随着web容器关闭而销毁,这就导致了这些java进程最终变成僵尸进程在系统中,蚕食着那微薄的内存空间了,而作为解决方案就是设置关闭钩子,以此当web容器关闭的时候,通过这个来关闭那些定时任务。或者就是每次关闭tomcat的时候,通过kill命令关闭那些仍然运行着的java进程。为了快速解决内存不足的问题,我选择了后者,因为毕竟项目并不多,先以简单的方式解决这个重要紧急的问题,通过kill然后再部署项目,结果发现,我们的项目的内存占用简直就是一丢丢呀,啊~,这是相当的幸福啊,看到还剩这么多内存的时候,我的心终于放下了,毕竟之前老是各个1-2个新奇,或者2个月左右挂掉,现在也已经逐渐稳定了。
感恩这段的经历,成为了我宝贵的财富,感恩提供文章的那些作者,造福了我们这些后辈,感恩遇到的一切。
参考文章:
https://blog.csdn.net/will_awoke/article/details/38338519?utm_source=tuicool&utm_medium=referral
https://www.cnblogs.com/sxdcgaq8080/p/10734752.html
linux服务器只部署了2个项目,却时常内存占满的问题解决路程
原文:https://www.cnblogs.com/Grace-is-enough/p/10890977.html