Java的四种引用
在Java语言中,除了原始数据类型的变量,其他所有都是所谓的引用类型,指向各种不同的对象,理解引用对于掌握Java对象生命周期和JVM内部相关机制非常有帮助。本文讲述了强引用、软引用、弱引用、幻象引用的区别以及一些具体使用场景,而且是配合ReferenceQueue使用。
强引用
我们平常典型编码0bject obj = new Object()中的obj就是强引用。通过关键字new创建的对象所关联用就是强引用。当IVM内存空间不足,JVM宁愿抛出OutOfMemoryError运行时错误(OOM),使程序异常终止,也不会靠随意回收具有强引用的"存活”对象来解决内存不足的问题。对于个普通的对象,如果没有其他的引用关系,只要超过了引用的作用域或者显式地将相应(强)引用赋值为null,就是可以被垃圾收集的了,具体回收时机还是要看垃圾收集策略。
软引用
软引用通过SoftReference类实现。软引 l用的生命周期比强引用短一些。只有当JVM认为内存不足时,才会去试图回收软引用指向的对象:即JVM 会确保在抛出OutOfMemoryError之前,清理软引用指向的对象。软用可以和一个引用队列(ReferenceQueue) 联合使用,如果软引用所引用的对象被垃圾回收器回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中。后续,我们可以调用ReferenceQueue的poll()方法来检查是否有它所关心的对象被回收。如果队列为空,将返回一个null,该方法返回队列中前面的一个Reference对象。
应用场景:软引用通常用来实现内存敏感的缓存。如果还有空闲内存,就可以暂时保留缓存,当内存不足时清理掉,这样就保证了使用缓存的同时,不会耗尽内存。
String str = new String("ABC"); //强引用
SoftReference<String> softReference = new SoftReference<>(str); //软引用
弱引用
弱引用通过WeakReference类实现。弱用的生命周期比软引用短。在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。于垃圾回收器是一个优先级很低的线程,因此不一定会很快回收弱引用的对象。弱引用可以和引用队列(ReferenceQueue) 联合使用,如果弱引用所引用的对象被垃圾回收,Java虛拟机就会把这个弱用加入到与之关联的引用队列中。
应用场景:弱应用同样可用于内存敏感的缓存。
在静态内部类中,经常会使用虚引用。例如:一个类发送网络请求,承担callback的静态内部类,则常以虚引用的方式来保存外部类(宿主类)的引用,当外部类需要被JVM回收时,不会因为网络请求没有及时回来,导致外部类不能被回收,引起内存泄漏
String str = new String("ABC"); //强引用
WeakReference<String> weakReference = new WeakReference<>(str); //弱引用
虚引用
特点:虚引用也叫幻象引用,通过PhantomReference类来实现。无法通过引用访问对象的任何属性或函数。幻象引用仅仅是提供了一种确保对象被finalize以后,做某些事情的机制。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。虚引用必须和引用队列(ReferenceQueue) 联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。
ReferenceQueue queue = new ReferenceQueue();
PhantomReference pr = new PhantomReference(object, queue);
程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。如果程序发现某个引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取一些程序行动。
应用场景:可用来跟踪对象被垃圾回收器回收的活动,当一个引用关联的对象被垃圾收集器回收之前会收到一系统通知。
String str = new String("ABC"); //强引用
ReferenceQueue<String> referenceQueue = new ReferenceQueue<>(); //引用队列
PhantomReference<String> phantomReference = new PhantomReference<String>(str,referenceQueue); //虚引用
ReferenceQueue
四种引用由强到弱分别是:强引用 > 软引用 > 弱引用 > 虚引用。
ReferenceQueue无实际存储结构,存储逻辑依赖于内部节点之间的关系来表达。存储关联的且被GC的软引用,弱引用以及虚引用。下面可以看一个示例:
NormalObject.java
package xpu.edu.tim;
public class NormalObject {
public String name;
public NormalObject(String name) {
this.name = name;
}
@Override
protected void finalize() throws Throwable {
System.out.println("Finalizing obj " + name);
}
}
NormalObjectWeakReference.java
package xpu.edu.tim;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.WeakReference;
public class NormalObjectWeakReference extends WeakReference<NormalObject> {
public String name;
public NormalObjectWeakReference(NormalObject normalObject, ReferenceQueue<NormalObject> q) {
super(normalObject, q);
this.name = normalObject.name;
}
@Override
protected void finalize() throws Throwable {
super.finalize();
System.out.println("Finalizing NormalObjectWeakReference "+ name);
}
}
ReferenceQueueTest.java
package xpu.edu.tim;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.concurrent.TimeUnit;
public class ReferenceQueueTest {
private static ReferenceQueue<NormalObject> rq = new ReferenceQueue<>();
private static void checkQueue(){
Reference<NormalObject> reference = null;
while ((reference = (Reference<NormalObject>) rq.poll()) != null){
if(reference != null){
System.out.println("In queue: "+ ((NormalObjectWeakReference)reference).name);
System.out.println("reference object: "+ reference.get());
}
}
}
public static void main(String[] args) throws InterruptedException {
ArrayList<WeakReference<NormalObject>> weakReferenceArrayList = new ArrayList<>();
for (int i = 0; i < 3; i++) {
weakReferenceArrayList.add(new NormalObjectWeakReference(new NormalObject("Weak " + i), rq));
System.out.println("Created weak:" + weakReferenceArrayList.get(i));
}
System.out.println("First time");
checkQueue();
System.gc();
TimeUnit.SECONDS.sleep(1);
System.out.println("Second time");
checkQueue();
System.out.println("Third time");
}
}
上面的示例是否有点复杂呢?看看下面这个也行:
package xpu.edu.tim;
import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.util.concurrent.TimeUnit;
public class EasyReferenceQueueDemo {
public static void main(String[] args) {
Object object = new Object();
ReferenceQueue referenceQueue = new ReferenceQueue<>();
PhantomReference<Object> phantomReference = new PhantomReference<>(object, referenceQueue);
object = null;
System.gc();
try {
TimeUnit.SECONDS.sleep(1); //给GC足够时间回收
Reference<Object> reference = referenceQueue.remove(2000L);
if(reference != null){
//TODO something
System.out.println("do something");
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
最终打印出do something,remove是一个阻塞方法,可以指定timeout, 或者选择一直阻塞。