内存泄漏和溢出

文章目录


前言

内存泄漏(Memory Leak)和内存溢出(Memory Overflow,通常称为 OOM)是软件开发中两个紧密相关但本质完全不同的概念。简单来说,内存泄漏是"该还的没还",而内存溢出是"借不到钱了"。

维度 内存泄漏 (Memory Leak) 内存溢出 (Out Of Memory - OOM)
本质 资源回收失败:无用对象因未被释放而一直占用内存。 资源分配失败:申请的内存超过了系统可用的上限。
表现 内存使用量缓慢、持续地增长(像慢性病)。 程序突然崩溃,抛出 OutOfMemoryError 异常。
因果关系 是导致内存溢出的常见诱因(长期泄漏最终耗尽内存)。 是内存泄漏的最终结果之一,但也可能由其他原因直接引起。
类比 租了房子不退,钥匙也不还,导致空房越来越少。 想租房子时,发现所有房子都被占用了,没房可租了。

一、内存泄漏

内存泄漏指的是程序在运行过程中,由于疏忽或错误,未能释放那些已经不再使用的内存。这些内存块虽然对程序已经无用,但因为某些原因(如被错误的引用持有)无法被垃圾回收器(GC)回收。

  • 常见原因:
    • 长生命周期对象持有短生命周期对象:例如,静态集合类(static List)一直往里添加对象,导致对象无法被回收。
    • 未关闭的资源:数据库连接(Connection)、文件流(InputStream)、网络连接等使用后未调用 close() 方法。
    • 内部类持有外部类引用:非静态内部类(如 Android 中的 Handler)隐式持有外部类(Activity)的引用,导致外部类无法销毁。
    • 监听器/回调未注销:注册了监听器但忘记注销,导致对象一直被引用。
  • 后果:程序运行越来越慢,内存占用越来越高,最终可能导致系统卡顿或触发 OOM。

二、内存溢出

内存溢出指的是程序在申请内存时,系统没有足够的内存空间供其使用。这通常发生在程序需要的内存超过了 JVM 配置的最大限制(如 -Xmx 参数),或者操作系统本身无法提供更多内存时。

  • 常见原因:
    • 内存泄漏的累积:这是最常见的原因,长期的泄漏最终耗尽了堆内存。
    • 加载数据量过大:例如一次性从数据库查询几百万条数据到内存中,或者读取一个超大文件到字节数组。
    • 配置过小:JVM 堆内存参数(-Xmx)设置得太小,不足以支撑业务运行。
    • 死循环或无限递归:导致栈空间不断被消耗(栈溢出 StackOverflowError)。
  • 后果:程序直接崩溃,抛出 java.lang.OutOfMemoryError 错误,服务中断。

如何排查与解决?

  • 排查内存泄漏:
    • 工具:使用内存分析工具,如 Java 的 MAT (Memory Analyzer Tool)、VisualVM、JProfiler。
    • 方法:监控内存使用曲线,如果内存随时间推移呈"锯齿状"上升且 GC 后不回落,极有可能是内存泄漏。通过工具分析堆转储文件(Heap Dump),找到"支配树"(Dominator Tree)中占用内存最大的对象及其引用链。
  • 解决内存泄漏:
    • 及时将不用的对象设为 null(虽然通常不需要,但在极端场景下有用)。
    • 使用 try-with-resources 语法自动关闭资源。
    • 避免使用静态集合类无限制地存储对象,或使用弱引用(WeakReference)。
    • 及时注销监听器和回调。
  • 解决内存溢出:
    • 应急:适当增加 JVM 内存参数(如 -Xmx4g),但这只是缓兵之计。
    • 优化代码:检查是否有一次性加载大数据的逻辑,改为分页或流式处理。
    • 修复根本问题:如果是由于内存泄漏导致的溢出,必须先修复泄漏点。
相关推荐
C+-C资深大佬3 分钟前
C++ 性能优化 专业详解
java·c++·性能优化
程序员老乔4 分钟前
Java 新纪元 — JDK 25 + Spring Boot 4 全栈实战(三):虚拟线程2.0,电商秒杀场景下的并发革命
java·开发语言·spring boot
weixin_404157686 分钟前
Java高级面试与工程实践问题集(四)
java·开发语言·面试
cyforkk13 分钟前
Spring AOP 核心揭秘:ProceedingJoinPoint 与反射机制详解
java·python·spring
2301_8101609517 分钟前
使用Flask快速搭建轻量级Web应用
jvm·数据库·python
无限进步_18 分钟前
【C++】单词反转算法详解:原地操作与边界处理
java·开发语言·c++·git·算法·github·visual studio
wyiyiyi20 分钟前
【线性代数】对偶空间与矩阵转置及矩阵分解(Java讲解)
java·线性代数·支持向量机·矩阵·数据分析
你这个代码我看不懂24 分钟前
磁盘的存储原理
java
PyAIGCMaster41 分钟前
开发了一个全自动接入wordpress的saas发文章的网站,记录一下如何实现,有需要的朋友联系。
java·开发语言·数据库
2201_7610801942 分钟前
Python上下文管理器(with语句)的原理与实践
jvm·数据库·python