在jvm中如何判断对象是生存还是死亡?
如何判断对象已死?
- 引用计数法:
给对象添加一个引用计数器,每当有地方引用时则加一,失效时减一,任何时刻计数器为零的对象是不可能再被使用的,
优点:引用计数法的实现简单判定效率高
缺点:难以解决对象间的相互循环引用问题 - 可达性分析算法:
通过一系列称为‘GC Roots’的对象作为起始点,从GC Roots开始向下搜索,搜索经过的路径称为引用链,当一个对象到GC Root没有任何引用链相连,则说明此对象是不可用的,是可被回收的。
Object1,2,3是仍然存活的对象,Object4,5,6是可回收对象 - 可作为GC Root的对象:
- 虚拟机栈(栈帧中的本地变量表)中引用的变量
- 方法区中类静态属性引用的对象
- 方法区中常量引用的对象
- 本地方法栈中JNI(Native方法)引用的对象
引用类型
不管是使用“引用计数法”还是“可达性分析算法”来判断对象是否存活都和“引用相关”。在一种应用场景中:有一类对象,当内存空间足够时,则保留在内存中,内存空间经过垃圾回收后内存还是不够则不是强引用的对象则可以进行抛弃,比如 很多系统的缓存功能就符合这种应用场景。
引用类型
- 强引用(Strong reference):代码中普遍存在的,类似于new的对象,只要强引用还存在,垃圾回收器永远不会收掉被引用的对象。
- 软引用(Soft Reference):用于还有用,但是并非必须的对象。会在系统将发生内存溢出异常之前,将对这些引用进行第二次回收,如果回收之后系统内存还是不够用,则抛出内存溢出异常,jdk使用SoftReference来实现
- 弱引用(Weak Reference):也用于描述非必须引用,但它的强度比软引用更弱一点,被弱引用关联的对象只能生存到下一次垃圾回收之前,不管当前系统内存是否够用都会被回收。使用WeakReference
- 虚引用(Phantom Reference):它是最弱的一种引用 ,虚引用的存在,完全不会对其生存时间构成影响,无法通过虚引用获得对象的实例,其存在的目的就是能够在这个对象被搜集器回收时收到一个系统通知。使用PhantomReference实现
注:四种引用的强度依次减弱
宣布对象死亡
要真正宣判一个对象死亡要经历两次标记过程
- 对象经历可达性分析后没有与GC Roots相连接的引用链,则进行一次标记和筛选。(筛选的条件是此对象是否有必要执行finalize方法。当对象没有覆盖finalize方法或该方法已被虚拟机执行过则没必要进行finalize方法)
- 在1筛选中,对象被判断为有必要执行finalize方法,则将对象放入F-Queue 队列中,并在稍后由虚拟机自动建立的,低优先级的Finalizer线程去执行对象的finalize()方法。这里是“执行该方法”,虚拟机并不保证该方法运行结束。稍后虚拟机对F-Queue进行第二次标记,对象唯一能逃脱被回收这种命运的办法是在执行finalize方法时和任意一个引用链上的对象关联即可,这时在进行第二次标记时会将对象重“即将回收的集合”中移除,对象获得新生。
注 需要注意的是finalize方法在对象的生命周期中最多执行一次,通过这种方式拯救对象是不可取的。
关注微信公众号一起阅读学习