Java内存泄漏分析和预防

1. 什么是内存泄漏?有什么危害

书面说法:

内存泄漏:对象已经没有被应用程序使用,但是垃圾回收器没办法移除它们,因为还在被引用着。
在Java中,内存泄漏就是存在一些被分配的对象,这些对象有下面两个特点,首先,这些对象是可达的,即在有向图中,存在通路可以与其相连其次这些对象是无用的,即程序以后不会再使用这些对象。如果对象回收不了,这些对象就可以判定为Java中的内存泄漏,它却占用内存。
偶尔一两次泄漏没太大影响,如果经常泄漏,达到一定质变机会引起内存溢出

2 Java内存泄漏引起的原因

内存泄漏是指无用对象(不再使用的对象)持续占有内存或无用对象的内存得不到及时释放,从而造成内存空间的浪费称为内存泄漏。内存泄露有时不严重且不易察觉,这样开发者就不知道存在内存泄露,但有时也会很严重,会提示你Out of memory。

我自己的理解:
长生命周期的对象持有短生命周期对象的引用就很可能发生内存泄漏,尽管短生命周期对象已经不再需要,但是因为长生命周期持有它的引用而导致不能被回收,这就是Java中内存泄漏的发生场景。如下图:
Java内存泄漏分析和预防

A对象引用B对象,A对象的生命周期(t1-t4)比B对象的生命周期(t2-t3)长。当B对象没有被应用程序使用之后,A对象仍然在引用着B对象。这样,垃圾回收器就没办法将B对象从内存中移除,从而导致内存问题,因为如果A引用更多这样的对象,那将有更多的未被引用对象存在,并消耗内存空间。

再极端一点。B对象也可能会持有许多其他的对象,那这些对象同样也不会被垃圾回收器回收。所有这些没在使用的对象将持续的消耗之前分配的内存空间。

再再极端一点,A对象是在永久代里,怎么办?

从预防内存泄漏上来讲:

最基本的建议就是尽早释放无用对象的引用,在确认一个对象无用后,将其所有引用显式的置为null。

面对已经发生的内存泄漏:

市场上已有几种专业检查Java 内存泄漏的工具,它们的基本工作原理大同小异,都是通过监测Java 程序运行时,所有对象的申请、释放等动作,将内存管理的所有信息进行统计、分析、可视化。

上一篇:学习使用Mendeley1


下一篇:Java内存泄漏分析系列之七:使用MAT的Histogram和Dominator Tree定位溢出源