这是和可达性分析法有关,该算法用作判断一个对象是否为垃圾对象,算法会从一个对象出发,从该对象的引用依次进行标记,所有被标记的对象形成一个类似树的结构,叫做引用链,而根节点就叫做GCRoot对象。在java堆中有多个GCRoot对象,在执行可达性算法后,对于没有被标记的对象,就会执行垃圾回收算法进行回收,该算法可以解决引用计数法中循环依赖的导致对象无法清除的问题。
由此引出来一个问题,如果一个对象没有被标记到,一定会被回收吗?
答案是不一定,因为Object类中提供了一个finalize方法,该方法给了对象一次机会,但也仅仅只有一次机会而已。当对象不可达(代表可回收)时,一旦发生GC垃圾回收,会先去判断对象是否执行了finalize方法。
第一种情况。如果未执行,则会首先执行finalize方法,如果我们在finalize方法中将当前对象和GCRoot对象关联,那么执行完该方法后,GC再去判断对象是否可达时,这个对象就变成可达,不会回收,如果这个对象还是不可达才会进行回收;
第二种情况,当该对象已经执行了finalize方法时,则垃圾回收算法会执行忽略这个方法,将这个对象回收,所以叫做有且只有一次机会可能躲避回收。
注意: finalize 方法只会被执行一次,如果第一次执行 finalize 方法此对象变成了可达确实不会回收,但如果对象再次被 GC,则会忽略 finalize 方法,对象会被回收!这一点切记!
那到底哪些对象可以作为GCRoot对象呢?
虚拟机栈(栈帧中的本地变量表)中引用的对象
本地方法栈中 JNI(即一般说的 Native 方法)引用的对象
方法区中类静态属性引用的对象
方法区中常量引用的对象