对象存活状态的判定以及死亡过程

关于对象的存活状态的判定可以使用两种算法。

第一种是引用计数算法,每一个对象都会有一个私有计数器,每当有一个地方引用他时,便会+1;相应的,有引用失效时,计数器就会-1,而当计数器变为0时,这个对象就是不可能再被使用的了。

第二种是可达性分析算法,从一个顶点对象(GC Roots)从上向下搜索,走过的路程被叫做引用链,,当有对象和GC Roots之间不可达时,这个或者这些对象就是不可用的了,即使这些对象有许多都是在被关联在一起。

下面是对象的死亡过程。

可以先看一段代码。

public class FinalizeEscapeGC {
    static FinalizeEscapeGC SAVE_HOOK=null;

    void islive(){
        System.out.println("yes,i am still alive :)");
    }

    @Override
    protected void finalize() throws Throwable {
        super.finalize();
        System.out.println("finalize method executed");
        FinalizeEscapeGC.SAVE_HOOK=this;
    }

    public static void main(String[] args) throws InterruptedException {
        SAVE_HOOK=new FinalizeEscapeGC();

        SAVE_HOOK=null;
        System.gc();
        Thread.sleep(500);
        if(SAVE_HOOK!=null){
            SAVE_HOOK.islive();
        }else{
            System.out.println("no ,i am dead :(");
        }

        SAVE_HOOK=null;
        System.gc();
        Thread.sleep(500);
        if(SAVE_HOOK!=null){
            SAVE_HOOK.islive();
        }else{
            System.out.println("no ,i am dead :(");
        }
    }
}

这是运行结果:

finalize method executed
yes,i am still alive :)
no ,i am dead :(

可以看到finalize()方法制只被用了一次,而且其中第一次回收的时候里面重新赋予了值,,而第二次会直接被回收掉。

在上面的两种算法中判定对象不可存活,当时对象真正的死亡也是要经历两次标记。用可达性分析算法来说,当发现对象没有和GC Roots相连接的可用链时,就会进行一次标记,并进行筛选,假如对象没有覆盖finalize()方法或者已经执行过finalize()方法了,就没有必要执行finalize()方法。

假如没有执行这个方法,它就会被放进一个队列中去,稍等一会GC会对这个队列进行小规模的二次标记,假如它完成了自救,就不用死亡,也就是重新和一个引用链的对象进行连接。没有的话,就会真的被回收掉。

这样一来就可以解释了上面的代码,在第一次被回收的时候进行了标记,还没有被真正的回收掉,也重新和一个引用链上的对象完成了链接,但是标记还在,之后又被赋予了空值,再次被回收,且只有第一次回收进行了finalize()方法,所以第二次回收的时候是被真正的回收掉了。

 

参考书籍:《深入理解JAVA虚拟机》

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值