finalization机制
JVM虚拟机再进行内存中的垃圾回收时,会先判断对象中的finalize()方法是否被重写,被重写的方法是否与GC Roots有直接或间接的联系,从而决定当前对象是否被回收,这就是JVM的finalization机制。
finalize()方法
finalize()方法允许在子类中被重写,用于在对象被回收时进行资源释放。
通常在这个方法中进行一些资源释放和清理的工作,比如关闭文件、套接字和数据库连接等。
finalize()方法由一个虚拟机自动创建的、低优先级的Finalizer线程触发待回收对象的finalize()方法执行,所以永远不要主动调用某个对象的finalize()方法,应该交给应圾回收机制调用,并且一个糟糕的 finalize()方法会影响系统GC的性能。
待回收对象的状态
由于finalize()方法的存在,虚拟机中的对象一般处于三种可能的状态:
如果从所有的根节点都无法访问到某个对象,说明对象己经不再使用了。一般来说,此对象需要被回收。但事实上,也并非是非死不可的,这时候它们暂时处于“缓刑”阶段。一个无法触及的对象有可能在某一个条件下“复活”自己,如果这样,那么对它的回收就是不合理的,为此,定义虚拟机中的对象可能的三种状态。如下:
- 可触及的:从根节点开始,可以到达这个对象。
- 可复活的:对象的所有引用都被释放,但是对象有可能在finalize()中复活。
- 不可触及的:对象的finalize()被调用,并且没有复活,那么就会进入不可触及状态。不可触及的对象不可能被复活,因为finalize()只会被调用一次。
以上3种状态中,是由于finalize()方法的存在,进行的区分。只有在对象不可触及时才可以被回收。
对象被回收的过程
判定一个对象。A是否可回收,至少要经历两次标记过程:
-
如果对象A到GCRoots没有引用链,则进行第一次标记。
-
进行筛选,判断此对象是否有必要执行finalize()方法I
①如果对象A没有重写finalize()方法,或者finalize()方法已经被虚拟机调用过,则虚拟机视为“没有必要执行”,A被判定为不可触及的。
②如果对象。A重写了finalize()方法,且还未执行过,那么A会被插入到F-Queue队列中,由一个虚拟机自动创建的、低优先级的Finalizer线程触发其finalize()方法执行。
③finalize()方法是对象逃脱死亡的最后机会,稍后GC会对F-Queue队列中的对象进行第二次标记。如果A在finalize()方法中与引用链上的任何一个对象建立了联系,那么在第二次标记时,A会被移出“即将回收”集合。之后,对象会再次出现没有引用存在的情况。在这个情况下,finalize方法不会被再次调用,对象会直接变成不可触及的状态,也就是说,一个对象的finalize方法只会被调用一次。