JAVA中的强软弱虚4中引用介绍,以及finalize什么时候被调用

1 篇文章 0 订阅

基本概念

  1. 强引用:通常我们通过new来创建一个新对象时返回的引用就是一个强引用,强引用是不会被GC回收的,JVM在内存不足时,会抛出OOM异常。

    一个对象若是通过一系列强引用可达,则这个对象就不会被回收。

  2. 软引用(SoftReference):在内存足够情况下,是不会被GC回收,只有在内存不够时才会被GC回收。

    若一个对象只通过软引用可达,这个对象在内存不足时会被回收,比弱引用强一点点。

    软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被垃圾回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中。软引用可用来实现内存敏感的高速缓存。

  3. 弱引用(WeakReference):发生GC就会被回收,不管内存是否足够。若一个对象只通过弱引用可达,那么发生GC时,这个对象就会被回收掉。不过,由于垃圾回收器是一个优先级很低的线程, 因此不一定会很快发现那些只具有弱引用的对象。

    弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。

    在ThreadLocal中有对弱引用的实现,为防止内存泄漏。

  4. 虚引用(PhantomReference):虚引用跟其他三种引用不同,他并不会对对象的回收有任何附加影响,只是用于被GC时跟踪一下回收状态。
    跟其他三个引用的不同还在于:虚引用必须和引用队列 (ReferenceQueue)联合使用。通过监听引用队列是否有引用,判断是否被GC回收。虚引用get方法返回的永远是NULL。
    虚引用可用于管理堆外内存,DirectByteBuffer类可用于管理堆外内存,DirectByteBuffer里面会创建一个Cleaner实例,而Cleaner就是PhantomReference的子类。

finalize拯救对象不被回收

在对象GC ROOTS不可达只通过弱引用可达只通过软引用可达并且内存不足时,通常情况下,在GC时是会被回收,回收过程并不是立即死亡,非死不可,还有机会拯救。
一个对象要被垃圾回收算法回收,至少要经历2次标记过程:在达到上面说的GC条件时,它将会被第一次标记并判断此对象是否有必要执行finalize()方法;当对象没有覆盖finalize()方法,或者finalize()方法已经被虚拟机调用过(finalize只会被执行一次),将被判断为没必要执行。
如果这个对象被判定为有必要执行finalize()方法,那么这个对象将会放置在一个叫做F-Queue的队列之中,并由虚拟机创建一个低优先级的Finalizer线程去执行它(该执行是指只会触发,也就是让finalize方法启动执行,并不会等待它执行完成,因为finalize可被程序员重写,对JVM来说是不可控的,也就是可能程序员在里面写死循环导致一直执行不完,也没法执行队列后面的方法)。

finalize()方法是对象逃脱死亡命运的最后一次机会,稍后GC将对F-Queue中的对象进行第二次小规模的标记,如果对象要在finalize()中成功拯救自己——只要重新加上强引用(如把自己(this关键字)赋值给某个类变量或者对象的成员变量),那么在第二次标记时它将被移除出“即将回收”的集合(F-Queue);因为每个对象的finalize只会被JVM执行一次,所以每个对象只有一次拯救自己的机会。

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值