查找内存中再也不使用的对象java
引用计数法算法
引用计数法就是若是一个对象没有被任何引用指向,则可视之为垃圾。这种方法的缺点就是不能检测到环的存在。code
2.根搜索算法对象
根搜索算法的基本思路就是经过一系列名为”GC Roots”的对象做为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链(Reference Chain),当一个对象到GC Roots没有任何引用链相连时,则证实此对象是不可用的。内存
引用计数法ci
下面经过一段代码来对比说明:虚拟机
public class MyObject { public Object ref = null; public static void main(String[] args) { MyObject myObject1 = new MyObject(); MyObject myObject2 = new MyObject(); myObject1.ref = myObject2; myObject2.ref = myObject1; myObject1 = null; myObject2 = null; } }
上述代码中myObject1和myObject2其实互相引用,他们的引用计数都为1,可是自己都是null,若是用引用计数法由于计数为1不会被GC回收,但他们自己为null,最终致使内存泄漏数学
若是采用的是引用计数算法:class
再回到前面代码GcDemo的main方法共分为6个步骤:变量
执行到Step 4,则GcObject实例1和实例2的引用计数都等于2。
接下来继续结果图:
到此,发现GcObject实例1和实例2的计数引用都不为0,那么若是采用的引用计数算法的话,那么这两个实例所占的内存将得不到释放,这便产生了内存泄露。
根搜索算法
这是目前主流的虚拟机都是采用GC Roots Tracing算法,好比Sun的Hotspot虚拟机即是采用该算法。 该算法的核心算法是从GC Roots对象做为起始点,利用数学中图论知识,图中可达对象即是存活对象,而不可达对象则是须要回收的垃圾内存。这里涉及两个概念,一是GC Roots,一是可达性。
那么能够做为GC Roots的对象(见下图):
关于可达性的对象,即是能与GC Roots构成连通图的对象,以下图:
从上图,reference一、reference二、reference3都是GC Roots,能够看出:
能够得出对象实例一、二、四、6都具备GC Roots可达性,也就是存活对象,不能被GC回收的对象。
而对于对象实例三、5直接虽然连通,但并无任何一个GC Roots与之相连,这即是GC Roots不可达的对象,这就是GC须要回收的垃圾对象。
到这里,相信你们应该能完全明白引用计数算法和根搜索算法的区别吧。 再回过头来看看最前面的实例,GcObject实例1和实例2虽然从引用计数虽然都不为0,但从根搜索算法来看,都是GC Roots不可达的对象。 总之,对于对象之间循环引用的状况,引用计数算法,则GC没法回收这两个对象,而根搜索算法则能够正确回收。