MyObject aRef = new MyObject(); SoftReference aSoftRef=new SoftReference(aRef); aRef = null; //现在只有一个软引用指向MyObject的这个对象, //如果这个对象还没有被回收,可以把他再次变为强引用 if(aSoftRef.get() != null) MyObject bRef = aSoftRef.get(); //这个时候MyObject这个对象又变成强引用 复制代码
boolean isEnqueued();通过程序或垃圾收集器来告知这个引用对象是否已经入队;
其中enqueue 和 isEnqueued 这两个方法涉及到引用队列,我们后面会讲到。这里就先不解释,留个印象就行。
软引用通过SoftReference类实现。软引用的生命周期比强引用短一些。只有当 JVM 认为内存不足时,才会去试图回收软引用指向的对象:即JVM 会确保在抛出 OutOfMemoryError 之前,清理软引用指向的对象。软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被垃圾回收器回收,Java虚拟机就会把这个软引用(注意是引用本身这个对象(就是Reference自己,并不是引用所引用的对象)加入到与之关联的引用队列中。后续,我们可以调用ReferenceQueue的poll()方法来检查是否有它所关心的对象被回收(因为在这个队列里面的引用所指向的对象都被回收了)。如果队列为空,将返回一个null,否则该方法返回队列中前面的一个Reference对象。
应用场景:软引用通常用来实现内存敏感的缓存。如果还有空闲内存,就可以暂时保留缓存,当内存不足时清理掉,这样就保证了使用缓存的同时,不会耗尽内存。
弱引用通过WeakReference类实现。 弱引用的生命周期比软引用短。在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。由于垃圾回收器是一个优先级很低的线程,因此不一定会很快回收弱引用的对象。弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中(和引用队列一起使用同上面的软引用)。
应用场景:弱应用同样可用于内存敏感的缓存。
幻象引用也叫虚引用,通过PhantomReference类来实现。无法通过虚引用访问对象的任何属性或函数。幻象引用仅仅是提供了一种确保对象被 finalize 以后,做某些事情的机制。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取一些程序行动(和引用队列一起使用同上面软引用跟弱引用)。
应用场景:可用来跟踪对象被垃圾回收器回收的活动,当一个虚引用关联的对象被垃圾收集器回收之前会收到一条系统通知
ReferenceQueue queue = new ReferenceQueue(); SoftReference ref=new SoftReference(aMyObject,queue); 复制代码
SoftReference ref = null; while ((ref = (EmployeeRef) q.poll()) != null) { // 清除ref } 复制代码
在强引用、软引用、弱引用、幻象引用的介绍我引用了一些其他博客下的评论,由于不好贴链接就只能声明一下吧。