垃圾收集器在对堆进行回收的时候第一件事情就是:
判断对象是否还存活
怎么判断对象是否还存活?
两种判断方式:引用计数器、可达性分析
引用计数器
实现简单、判定效率也挺高的
给对象头中添加一个引用计数器,当有一个地方引用该对象的时候,引用计数器加一;引用失效的时候减一。当引用计数器为零的时候,该对象便不可能在被使用。
当下非常火热的Python就是使用的这个判断方式。
可达性分析
java没有选用引用计数器来判断对象是否存活,是因为它很难解决对象之间的相互循环引用的问题。
可达性分析是通过一系列的“GC Roots”的对象作为起始点,然后向下进行搜索,搜索走过的路径称为引用链,当一个对象没有到“GC Roots”没有任何引用链的时候,则证明该对象不可达(即此对象是不可用的)
在java中,可以作为“GC Roots”对象的有:
- java虚拟机栈中引用的对象
- 本地方法栈中的Native方法引用的对象
- 方法区中的静态属性引用的对象
- 方法区中常量引用的对象
可达性分析之后并不能马上决定对象的生死,有一个二次标记过程。
我们创建的类都是继承超类Object的,Object类中有一个finalize(),它可能会让该类起死回生。
整个过程:在可达性分析之后,会对不可达对象进行第一次标记,然后进行一次筛选
筛选的规则是:
1 该对象有无重写finalize()
2 如果重写,finalize()是否已经被执行
如果该对象没有重写finalize()或者已经被执行过,则该对象可以被回收了。否则会将该对象放在一个叫做F-Queue的队列中,然后GC会对该队列进行第二次小规模的标记,如果对象在finalize()中拯救自己–只要与引用链上的对象建立关系就行。如果没有逃脱就会被回收