[面试题]java程序内存泄漏怎么排查

5 篇文章 1 订阅

上文提到面试碰到CPU占满时的问题解决,决心把内存泄漏的问题也一起看一看,如果有更好的方案,评论区请指教。

首先了解几个命令

pId 为程序id

# 获取当前linux上部署的所有java程序 pid和运行占有资源情况(俩个都可以)
ps -aux|grep java
top $(ps -e | grep java | awk '{print $1}' | sed 's/^/-p/')
#获取某个java程序所有线程资源占用情况
top -Hp [pId]
# 获取某个java程序的线程执行情况(会展示在哪个类的哪一行)
jstack   [pId]
# 获取某个java程序的gc情况
jstat -gcutil  [pId] 
# 获取某个java程序的gc情况 每秒获取一次 获取10次
jstat -gcutil  [pId]  1000 10
# 将某个java程序的heap(堆)情况写入heap.hprof文件
 jmap -dump:format=b,file=heap.hprof  [pId]

怎么判断当前程序有没有出现内存溢出

模拟代码

ps: 模拟代码大概要执行10分钟才会溢出,可以加入多个String多个集合实现快些。

import java.util.ArrayList;
import java.util.List;

/**
 * @Author 打工人
 * @Date 2022/4/4 下午9:14
 * @Version 1.0
 * @Remarks
 */
public class SystemOOMTest {

    public static final String SYSTEM_OOMTEST_STR = "SYSTEM_OOMTEST_STR";
    public static void main(String[] args) {
        int i = 0;
        String thisStr;
        List<String> oomObjs = new ArrayList<String>();
        while (true){
            thisStr = SYSTEM_OOMTEST_STR + i++;
            oomObjs.add(thisStr);
            try {
                Thread.sleep(100L);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(thisStr);
        }
    }

}

模拟步骤

jvm初始和最大内存设置2M运行

javac SystemOOMTest2.java
java -Xms2M -Xmx2M SystemOOMTest2

判断依据

每10秒展示一次gc 获取10次

jstat -gcutil 13869 10000 10

请添加图片描述
想了解图中每一项参数点这里
E为Edan区,新创建的string对象都会放在该区,图中展示的是占用比例。
jvm的对象创建机制为先存储Edan区(大对象会直接存储在Old区),存满后,执行YGC,没有被GC到的对象复制到S0或者S1区,都存满还是没有可用内存的话将S区的对象存储在O(Old)区,Old区存满之后执行FGC(FullGC)。

java程序内存溢出判断的大依据为FGC的次数,如果FGC次数很频繁且每次GC完释放的内存很少就有可能是出现了内存溢出。(为什么是有可能呢,因为也有情况是内存确实不够用)

出现内存溢出怎么办

最原始的方法

将java程序的堆占用情况输出到heap.hprof文件中,然后使用解析工具进行解析,解析完成之后查看实例占用多的类。然后进入代码分析,查看为什么没有被回收。
format=b:表示生成二进制类型的dump文件
PS: 执行下面语句jvm会执行一次FGC,并且会造成程序挂起(线上慎用)。文件大的话在解析时也会很慢。

 jmap -dump:format=b,file=heap.hprof [pId]

使用JProfiler解析hprof文件

在这里插入图片描述

在线dump文件分析网站https://console.heapdump.cn/

在这里插入图片描述

MAT

楼主最开始也尝试的这个工具,不过 MAC上面安装的1.8.0版本的,打开解析面板是空白的。

第二种方案

最好能在本地复盘。通过查看线上日志,和近期的发版内容,用户增量一起结合起来看。毕竟线上真遇见了,堆内存占用会很大,在写出的时候java程序会挂起,影响运行。

代码层面最好避免这种情况,使用完的对象和集合要确定会不会被jvm正常GC掉。

PS:堆中的实例对象GC依据为 当前实例对象还有无地方引用。

总结

  1. 定位到内存占用过高的java程序pid
  2. 查看java程序的gc情况
  3. 如果FGC次数很频繁 导出一份heap数据
  4. 使用解析工具解析heap

其他:注意代码规范,使用完的实例对象要确保能被回收掉。

希望本文可以帮到你。

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值