内存泄露排查

    技术2022-07-11  94

    概念

    内存泄露(Memory Leak):意思就是你用资源的时候为他开辟了一段空间,当你用完时忘记释放资源了,这时内存还被占用着,一次没关系,但是内存泄漏次数多了就会导致内存溢出

    内存溢出(Out Of Memory---OOM):系统已经不能再分配出你所需要的空间,比如你需要100M的空间,系统只剩90M了,这就叫内存溢出

    常见问题现象

    后台系统,经过压力测试之后,或者生产程序运行一段时间后,进程占用的操作系统内存比例一直居高不下。怀疑系统可能存在内存泄漏。

    常用工具及命令:visualVM, MAT, jmap, top

    排查思路:

    使用top命令查看问题进程,查看cpu,内存消耗情况   top -p 12121212

    查看该进程的启动参数等相关情况,判断参数设置是否合理  ps -aux |grep -v grep|grep 28990

    查看内存使用情况    jmap -heap 12121212

    查看内存实时使用情况,用visualVM远程连接程序(visualVM远程连接进程),查看使用趋势……

    经过以上几步如果已经确定是内存泄露,则需要导出heap dump进一步分析,导出方式有visualVM,jmap

    jmap -dump:file=/data/server/test_ops/a.hprof 22678

    jmap -dump:live,format=b,file=/data/server/test_ops/a.hprof 5889

    jmap -histo:live 5889

    heap dump分析,用MAT, visualVM结合起来分析

    不健壮代码的特征及解决办法 

    1、尽早释放无用对象的引用。好的办法是使用临时变量的时候,让引用变量在退出活动域后,自动设置为null,暗示垃圾收集器来收集该 对象,防止发生内存泄露。对于仍然有指针指向的实例,jvm就不会回收该资源,因为垃圾回收会将值为null的对象作为垃圾,提高GC回收机制效率; 

    2、我们的程序里不可避免大量使用字符串处理,避免使用String,应大量使用StringBuffer,每一个String对象都得独立占用内存一块区域;例如 

    3、尽量少用静态变量,因为静态变量是全局的,GC不会回收的; 

    4、避免集中创建对象尤其是大对象,JVM会突然需要大量内存,这时必然会触发GC优化系统内存环境;显示的声明数组空间,而且申请数量还极大。 

    5、尽量运用对象池技术以提高系统性能;生命周期长的对象拥有生命周期短的对象时容易引发内存泄漏,例如大集合对象拥有大数据量的业务对象的时候,可以考虑分块进行处理,然后解决一块释放一块的策略。 

    6、不要在经常调用的方法中创建对象,尤其是忌讳在循环中创建对象。可以适当的使用hashtable,vector 创建一组对象容器,然后从容器中去取那些对象,而不用每次new之后又丢弃 

    7、一般都是发生在开启大型文件或跟数据库一次拿了太多的数据,造成 Out Of Memory Error 的状况,这时就大概要计算一下数据量的最大值是多少,并且设定所需最小及最大的内存空间值

    Processed: 0.013, SQL: 9