下面来分析一下与Reference有关的sun.misc.Cleaner和java.lang.ref.Finalizer。
先来看Finalizer,Finalizer扩展了Reference,故其本质也是一个Reference,与SoftReference、WeakReference、PhantomReference不同的是gc遇到FinalReference后,并不会回收其referent,而只是入队,所入队列就是下图的queue,FinalizerThread会处理该队列。
FinalizerThread的主要逻辑:
runFinalizer方法:
这里还有一个隐藏问题,就是Finalizer对象由谁来维护?SoftReference是由我们自己来维护的,比如充当缓存(map)中的value,Finalizer是由其自身维护的,在Finalizer中有一个unfinalized静态字段,该字段用来存放Finalizer实例,这样就可以保证Finalizer在执行referent的finalize方法前不会被gc回收。
再来看一下Cleaner,cleaner扩展了PhantomReference,经过之前的讨论我们知道,PhantomReference只适合做对象回收的跟踪,而不能做对象的存取,Cleaner所需要的就是跟踪到对象(比如ByteBuffer)被gc回收后,做出相应逻辑处理。
之前提到过,clean方法会在HandlerThread中被调用,并且Cleaner不会入队。