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

本文详细介绍了如何通过命令行工具(如`jstat`, `jmap`, `jstack`)诊断Java内存泄漏,包括模拟内存溢出场景、观察GC行为,以及使用JProfiler和MAT工具进行深入分析。关键步骤包括设置初始和最大内存,模拟内存消耗,观察FGC频率和释放内存情况。

上文提到面试碰到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

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

希望本文可以帮到你。

### Java 内存泄漏常见面试题及解答 #### 什么是Java内存泄漏Java中的内存泄漏是指程序在运行过程中,对象不再被使用,但由于某些原因未能被垃圾回收器(GC)回收,导致内存资源的浪费。这种情况在长期运行的应用中尤为严重,可能导致OutOfMemoryError[^1]。 #### Java内存泄漏和C++内存泄漏的区别是什么? 在C++中,内存泄漏通常是由于程序员忘记手动释放不再使用的内存而导致的;而在Java中,内存泄漏主要是由于对象仍然被引用而无法被GC回收。Java的垃圾回收机制自动管理内存,因此内存泄漏更多是逻辑上的问题,而不是忘记释放内存[^1]。 #### Java内存泄漏的常见原因有哪些? - **长生命周期的对象持有短生命周期对象的引用**:例如静态集合类中持续添加对象而不移除,导致这些对象无法被回收。 - **缓存未清理**:缓存对象没有设置过期策略或容量限制,导致缓存无限增长。 - **监听器和回调未注销**:如事件监听器、观察者模式中的回调函数未及时取消注册。 - **内部类持有外部类引用**:非静态内部类(如匿名内部类)隐式持有外部类的引用,可能导致外部类无法被回收。 - **线程局部变量(ThreadLocal)使用不当**:如果ThreadLocal变量没有正确清除,可能会导致线程池中的线程长时间存活并持有不必要的对象引用[^1]。 #### 如何排查Java内存泄漏? - **使用VisualVM或JConsole等工具**:这些工具可以帮助监控堆内存的使用情况,并分析对象的引用链。 - **生成Heap Dump文件**:通过jmap命令生成堆转储文件,然后使用MAT(Memory Analyzer Tool)进行分析,找出内存占用最高的对象及其引用关系。 - **检查GC日志**:启用GC日志记录,分析Full GC的频率和内存回收效果,判断是否存在内存泄漏的迹象。 - **代码审查**:检查是否有不合理的对象引用、未关闭的资源句柄等问题。 #### Java内存泄漏的解决方案有哪些? - **合理设计对象生命周期**:确保对象在使用完毕后能够被及时释放,避免不必要的强引用。 - **使用弱引用(WeakHashMap)**:对于需要缓存的对象,可以考虑使用`WeakHashMap`,这样当键对象不再被引用时,对应的条目会自动从Map中移除。 - **及时关闭资源**:如IO流、数据库连接等资源应在finally块中关闭,确保不会因为异常而泄露。 - **优化缓存策略**:为缓存设置合理的过期时间和最大容量,防止缓存无限增长。 - **清理ThreadLocal变量**:在线程执行完毕后,调用`remove()`方法清除ThreadLocal变量,避免线程池中的线程持有无效的ThreadLocal实例。 #### 示例:如何使用WeakHashMap避免内存泄漏? ```java import java.util.WeakHashMap; public class CacheExample { private static final WeakHashMap<Key, Value> cache = new WeakHashMap<>(); public static void addToCache(Key key, Value value) { cache.put(key, value); } public static Value getFromCache(Key key) { return cache.get(key); } } ``` 在这个例子中,`Key`对象作为弱引用存储在`WeakHashMap`中。当`Key`对象不再被其他地方引用时,它将被GC回收,对应的`Value`也会从缓存中移除,从而避免了内存泄漏。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值