说说内存泄漏的常见场景和排查方案?

内存泄漏是指程序中某些对象不再被使用,但由于仍然被引用,垃圾回收器无法回收这些对象,导致内存被持续占用的问题。

PS:这就像桶破了个洞,水慢慢漏出,最终桶里的水会越来越少。

内存泄漏 ≠ 内存溢出

这里需要注意的内存泄漏和内存溢出是不同的内存溢出是指程序在申请内存时,系统无法提供足够的可用内存空间以满足需求,导致程序崩溃或异常。这就像往一个固定容量的桶里倒水,当水超过桶的容量时就会溢出。

所以说,长期的内存泄漏最终会导致内存溢出

内存泄漏常见场景

内存泄漏的常见场景如下:

  1. ThreadLocal 使用不当:线程池中未清理 ThreadLocal,示例代码如下:
java 复制代码
// ThreadLocal未 remove 导致泄漏
private static ThreadLocal<BigObject> threadLocal = new ThreadLocal<>();
public void processRequest() {
    threadLocal.set(new BigObject());
    // 使用后未调用 threadLocal.remove()
}
  1. 未关闭资源:文件流、数据库连接等未关闭,示例代码如下:
java 复制代码
// 文件流未关闭导致泄漏
public void fileLeak() throws Exception {
FileInputStream fis = new FileInputStream("largeFile.txt");
// 忘记调用 fis.close()
}
  1. 监听器未注销:注册的事件监听器未移除,示例代码如下:
java 复制代码
public class EventSource {
    private List<EventListener> listeners = new ArrayList<>();
    public void addListener(EventListener listener) {
        listeners.add(listener); // 缺少 removeListener方法
    }
}
  1. 静态集合滥用:静态集合长期持有对象引用,示例代码如下:
java 复制代码
// 静态集合导致内存泄漏
private static List<Object> cache = new ArrayList<>();
public void addToCache(Object obj) {
cache.add(obj); // 添加后从不移除
}
  1. 不正确的equals/hashCode:导致 HashMap 无法正确识别相同对象,示例代码如下:
java 复制代码
// 未重写 equals/hashCode 导致 HashMap 内存泄漏
Map<Person, Integer> map = new HashMap<>();
for(int i=0; i<100; i++) {
    map.put(new Person("jon"), 1); // 每次都被视为新对象
}

内存泄漏排查

内存泄漏的排查步骤如下:

  1. 监控内存使用趋势
    • 使用 jconsole 或 jvisualvm 观察内存是否持续增长。
    • 关注 Full GC 后内存是否回落。
  2. 堆转储分析
    • 使用 Eclipse MAT 分析堆转储文件。
    • 查看 Dominator Tree 找出占用内存最多的对象。
    • 检查 Reference Chain 定位阻止回收的引用。
  3. 代码审查
    • 静态集合的使用情况。
    • 资源关闭逻辑(try-with-resources)。
    • 监听器的注销机制。
    • ThreadLocal 的 remove 调用。

小结

内存泄漏是指程序中某些对象不再被使用,但由于仍然被引用,垃圾回收器无法回收这些对象 ,导致内存被持续占用的问题。长期的内存泄漏会导致内存溢出问题,但内存泄漏不等于内存溢出,这点面试的时候一定要注意。

本文已收录到我的面试小站 www.javacn.site,其中包含的内容有:场景题、SpringAI、SpringAIAlibaba、并发编程、MySQL、Redis、Spring、Spring MVC、Spring Boot、Spring Cloud、MyBatis、JVM、设计模式、消息队列、Dify、Coze、AI常见面试题等。

相关推荐
JAVA面经实录9174 小时前
Java企业级工程化·终极完整版背诵手册(无遗漏、全覆盖、面试+落地通用)
java·开发语言·面试
周杰伦fans5 小时前
AutoCAD .NET 二次开发:深入理解 EntityJig 的工作原理与正确实现
开发语言·.net
许彰午6 小时前
CacheSQL(二):主从复制——OpLog 环形缓冲区与故障自动恢复
java·数据库·缓存
小程故事多_807 小时前
[大模型面试系列] 多轮对话 Agent 设计实战(含窗口优化 + 工具调用精髓)
人工智能·面试·职场和发展
Bat U7 小时前
JavaEE|多线程初阶(七)
java·开发语言
谭欣辰7 小时前
C++ 排列组合完整指南
开发语言·c++·算法
foundbug9998 小时前
自适应滤除直达波干扰的MATLAB实现
开发语言·算法·matlab
XDH_CS8 小时前
MySQL 8.0 安装与 MySQL Workbench 使用全流程(超详细教程)
开发语言·数据库·mysql
小短腿的代码世界9 小时前
Qt实时盈亏计算深度解析:从持仓数据到动态盈亏展示
开发语言·qt