1、如何判断对象已死
1、引用记数法
给对象添加一个引用计数器,每当有一个对象引用时,计数器值就加一;当引用失效时,计数器指就减一;任何时刻计数器为零的对象就是不可能在被使用的。
2、可达性分析
这个算法的基本思路就是通过一系列的称为GCRoot的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路经称为引用链,(reference Chain)当一个对象到GCRoot没有任何引用链相连接时,则证明这个对象是不可使用的。
在java语言中,可以作为GCRoot的对象包含以下几种:
虚拟机栈中引用的对象
方法区中类静态属性引用的对象
方法区中常量引用的对象
本地方法栈中JNI(一般说的Native方法)引用的对象
字符串常量池中的引用(Srtring Table)
所有被同步锁持有的对象,比如被synchronize修饰的对象
类型为引用类型的静态变量
2、对象真的死了吗?
要真正宣告一个对象死亡,至少要经历两次标记过程:如果对象在进行可达性分析后发现没有GCRoots相连接的引用链,那它将被第一次标记并且进行一次筛选,筛选的条件是此对象是否有必要执行finalize()反法。或者finalize()方法已经被虚拟机调用过,虚拟机将这两种情况都视为没必要执行。
如果这个对象被判断为有必要执行finalize()方法,那么对象将会被放置在一个叫做F-queue的队列中,并且在稍后由一个虚拟机自动建立的、低优先级的Finalize线程去执行它。
finalize()方法是对象逃脱死亡的最后一次命运,稍后Gc将对F-queue中对象进行第二次小规模标记,如果对象要在finlize()中成功拯救自己----只要重新与引用链上的任何一个对象建立联系即可,譬如把自己赋值给某个变量或者对象的成员变量,那在第二次标记时他将被移除‘即将回收’的集合;如果对象这时候还没有逃脱,那基本上他就是真的被回收了。
3、回收方法区
很多人认为方法区(或者HotSpot虚拟机中的永久代)是没有垃圾收集的
永久代的垃圾回收主要分为两个部分:废弃常量和无用的类
类需要满足3个条件才能算是‘无用的类’:
1、该类所有的实例都已经被回收,也就是java堆中不存在该类的人呢和实例。
2、加载该类的classLoader已经被回收
3、该类对应的java.long.Class对象没有在任何地方被引用,无法在任何地方通过反射访问该类的方法