在Java中有四种引用类型,他们是:强引用(Strong Reference),软引用(Soft Reference),弱引用(Weak Reference) 和 虚引用(Phantom Reference)。
一.四种引用类型的解释:
类型 |
目的 |
作用 |
触发GC条件 |
实现类 |
强引用 |
普通引用类型,只要对象的引用是强引用,他们就不会被垃圾收集 |
普通引用 |
任何对象如果不是强引用都可以被垃圾收集 |
默认类型 |
软引用 |
在内存足够的时候,对象不会被垃圾收集 |
为了保证即使对象没有任何引用指向它的时候也不会被垃圾收集,防止有引用再次指向这个对象 |
在第一次GC后,JVM需要回收更多的空间 |
java.lang.ref.SoftReference |
弱引用 |
在对象可触及的状态下不会被垃圾收集 |
如果对象不再被引用会被自动回收 |
GC后对象只有弱引用 |
java.lang.ref.WeakReference |
虚引用 |
让你可以清理已经执行finalize方法,但是还没有回收内存的对象 |
特殊清理 |
finalize方法执行之后 |
java.lang.ref.PhantomReference |
StrongReference
这个不用多讲了,这是java默认的引用类型,如果不特意使用java.lang.ref下的类,那么程序中的所有引用都是强引用。有强引用存在的对象永远都不会被gc收集,所以在内存不够用时,JVM宁愿抛出OutOfMemoryError这样的错误,也不愿意将强引用对象进行回收。
SoftReference
软引用不会保证对象一定不会被回收,只能最大可能保证。如果内存有剩余,那么软引用对象不会被回收,如果内存不足,那么gc会回收软引用对象。所以这种特性可以用来实现缓存技术。软引用要用java.lang.ref.SoftReference来实现。
public class SoftTest{
public static void main(String[] args) {
Object ref = new Object();//ref是Object对象的强引用
//将一个软引用指向对象,此时Object对象有两个引用
SoftReference
ref = null;//去除对象的强引用
System.gc();//gc只有在内存不足是才会回收软引用对象
}
}
WeakReference
除了通过java.lang.ref.WeakReference来使用弱引用,WeakHashMap同样也利用了弱引用。
和软引用不同的是,弱引用一定会被gc回收,不管内存是否不足。
public class WeakTest{
public static void main(String[] args) {
Object ref = new Object();//ref是Object对象的强引用
//将一个弱引用指向对象,此时Object对象有两个引用
WeakReference
ref = null;//去除对象的强引用
System.gc();//gc对弱引用对象进行回收
}
}
PhantomReference
幽灵引用,也叫虚引用。java.lang.ref.PhantomReference类中只有一个方法get(),而且几乎没有实现,只是返回null。而且这个类只有一个构造器(软引用和弱引用均有两个构造器):
public PhantomReference(T referent, ReferenceQueue super T> q) {
super(referent, q);
}
也就是说,幽灵引用只能与ReferenceQueue(后面会提到这个类)一起使用。如果一个对象仅有幽灵引用,那么它就像没有任何引用一样,在任何时候都可能被gc回收。幽灵引用主要用来跟踪对象被垃圾回收的活动。
public class PhantomTest{
public static void main(String[] args) {
Object ref = new Object();//ref是Object对象的强引用
//将一个幽灵引用指向对象,PhantomReference必须与ReferenceQueue一同使用
PhantomReference
System.out.println(pf.get());
}
}
ReferenceQueue
如果一个对象只有软引用、弱引用或者幽灵引用,gc在回收对象时,JVM会自动将其引用放入一个ReferenceQueue中。WeakHashMap就是利用了ReferenceQueue来实现清除没有强引用Entry的。将上面的弱引用例子稍微改一下:
public class ReferenceQueueTest{
public static void main(String[] args) {
Object ref = new Object();//ref是Object对象的强引用
System.out.println(ref);
ReferenceQueue
WeakReference
System.out.println(sf);
ref = null;//去除对象的强引用,在这里加个断点,进行调试
System.gc();//gc对弱引用对象进行回收
System.out.println(rq.poll());
}
}
注意,运行这个程序需要用debug模式进行调试,在上面说明的地方加个断点。如果直接运行,结果很可能只是个null,用debug调试的话会看到输出的是弱引用的地址。程序输出的sf结果和rq.poll()结果相同。