在文章 JVM如何判断对象"死"和"活"中,介绍了如何判断对象生存与否,其中说到了根搜索算法,是否与GC Roots有通路来判断对象的生死属性。那么,java虚拟机发送内存泄露的原因是什么呢,主要有以下两个特点:
(1)被分配的对象是可达的,即在有向图中,与GC Roots是有通路的;
(2)但该对象在以后的程序中不再使用,即对象是无用的。
满足了上面两个特点,就可以判定为内存泄露,这些对象不会被GC回收,但仍占用着内存。
举个简单的例子如下:
package cn.com.yy;
import java.util.Vector;
/**
* 简单的内存泄露例子
* @author yy
* @time 2014-10-3 上午10:41:42
*/
public class MemoryLeakTest {
public static void main(String[] args) {
Vector v = new Vector(100000000);
for(int i=0;i<v.size();i++){
Object o = new Object();
v.add(o);
o = null;
}
}
}
执行该程序,结果如下:
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at java.util.Vector.<init>(Vector.java:131)
at java.util.Vector.<init>(Vector.java:144)
at cn.com.yy.MemoryLeakTest.main(MemoryLeakTest.java:13)
解析:
在for循环中,循环申请Object对象,每申请一个放入Vector中,然后把Object对象置空。此时,对于Object对象来说,与GC Roots是有通路的,因为由Vector对象引用,但在Object对象置空后,Object对象又没有用途了,所以对于所有的Object对象来说,与GC Roots有通路,但是又没有用,所以造成了堆内存泄露。