1 引用类型背景
在jdk 1.2以前,java中的引用的定义很传统:如果 reference
类型的数据中存储的数值代表的是另外一块内存的起始地址,就称这块内存代表着一个引用。
上面的定义很纯粹,但是太过狭隘,一个对象在这种定义下只有被引用或者没有被引用两种状态,对于如何描述一些“食之无味,弃之可惜”的对象就显得无能为力。我们希望能描述这样一类对象:
- 当内存空间还足够时,则能保留在内存之中
- 如果内存空间在进行垃圾收集后还是非常紧张,则可以抛弃这些对象
2 四种引用类型
在java1.2之后,java对引用的概念进行了扩充,将引用分为强引用、软引用、弱引用、虚引用,这4种引用强度依次逐渐减弱。
- 强引用(Strong Reference):在程序代码之中普遍存在的,类似
Object obj = new Object()
这类的引用,只要强引用还存在,垃圾收集器永远不会回收掉被引用的对象 - 软引用(Soft Reference):用来描述一些还有用但并非必需的对象。对于软引用关联着的对象,在系统将要发生内存溢出异常之前,将会把这些对象列进回收范围之中进行第二次回收。如果这次回收还没有足够的内存才会抛出内存溢出异常
- 弱引用(Weak Reference):用来描述非必需对象,但是它的强度比软引用更弱一些,被弱引用关联的对象只能生存到下一次垃圾收集发生之前。当垃圾收集器工作时,无论当前内存是否足够,都会回收掉只被弱引用关联的对象
- 虚引用(Phantom Reference):它是最弱的一种引用关系。一个对象是否有虚引用存在,完全不会对其生存时间构成影响,也无法通过虚引用来取得一个对象实例。为一个对象设置虚引用关联的唯一目的就是能在这个对象被收集器回收时收到一个系统通知
需要注意的是:
在Java开发时,你可能会了解到因为 SoftReference
软引用会在内存溢出的时候才会被回收,使用软引用作为缓存,但这在Android开发中是不被推荐的,因为Android的JVM和Java不完全相同可能会更激进一些。
在Android开发中更推荐使用 LruCache
作为缓存。
3 怎么知道引用要被回收
在这里我们可以将四种引用粗略的再区分一下,分为比较强的引用和比较弱的引用:
这里最主要说明比较弱的引用。
我们看软引用、弱引用和虚引用的源码时,可以发现它们有一个共同特征:构造中有需要传入一个参数 ReferenceQueue
:
SoftReference.java
public class SoftReference<T> extends Reference<T> {
public SoftReference(T reference) { ... }
public SoftReference(T reference, ReferenceQueue<? super T> q) { ... }
}
WeakReference.java
public class WeakReference<T> extends Reference<T> {
public WeakReference(T reference) { ... }
public WeakReference(T reference, ReferenceQueue<? super T> q) { ... }
}
PhantomReference.java
public class PhantomReference<T> extends Reference<T> {
public PhantomReference(T reference, ReferenceQueue<? super T> q) { ... }
}
那么 ReferenceQueue
是用来干嘛的?
如果三种比较弱的引用持有的 T reference
没有被其他强引用所引用(即被比弱一点的引用所引用到),在GC之前,会先把引用(这里指的是 SoftReference
、WeakReference
、PhantomReference
引用,而不是它们持有的引用 T reference
)添加到 ReferenceQueue
队列中,如果在队列中能知道 ReferenceQueue
有引用,那也就能说明 T reference
被垃圾回收了。
通过 ReferenceQueue
我们就可以很可靠的确认 T reference
是否被回收,这种方式相比 finalized()
判断引用是否被回收更可靠