Java 四种引用总结(强引用、软引用、弱引用、虚引用)
2019-04-28 本文已影响0人
jqboooo
1、强引用(StrongReference)
也就是我们new出来的对象当内存不足的时候,JVM宁可出现OutOfMemory错误停止,也需要进行保存,并且不会将此空间回收(永远不会被GC回收);
例如:
Object obj = new Object();
2、软引用(SoftReference)
内存不足时回收,存放一些重要性不是很强又不能随便让清除的对象,比如图片切换到后台不需要马上显示了.
软引用可用来实现内存敏感的高速缓存。
Object obj = new Object(); //强引用
ReferenceQueue<Object> referenceQueuee = new ReferenceQueue<>(); //引用队列
// 软引用和引用队列联合使用,如果软引用所引用的对象被垃圾回收器回收,JVM就会把这个软引用加入到引用队列中
SoftReference softReference = new SoftReference(str, referenceQueuee);
当内存不足时,等价于:
if(JVM.内存不足()){
obj = null;
System.gc();
}
3、弱引用(WeakReference)
第一次扫到了,就标记下来,第二次扫到就直接回收。
@Test
public void testWeakReference() throws InterruptedException {
ReferenceQueue<Object> referenceQueuee = new ReferenceQueue<>();
Object weakObject = new Object();
//弱引用
WeakReference weakReference = new WeakReference(weakObject, referenceQueuee);
System.out.println("WeakReference:" + weakReference.get());
System.out.println("referenceQueuee:" + referenceQueuee.poll());
weakObject = null;
System.gc();
Thread.sleep(2000);
System.out.println("WeakReference:" + weakReference.get());
System.out.println("referenceQueuee:" + referenceQueuee.poll());
}
测试的结果:
WeakReference:java.lang.Object@694f9431
referenceQueuee:null
WeakReference:null
referenceQueuee:java.lang.ref.WeakReference@f2a0b8e
4、虚引用(PhantomReference)
幽灵、幻影引用,不对对象生命周期造成任何影响,用于跟踪GC的回收通知。
功能:不会影响到对象的生命周期,但是能让程序员知道该对象什么时候被回收了。
使用虚引用的目的就是为了得知对象被GC的时机,所以可以利用虚引用来进行销毁前的一些操作,比如说资源释放等。这个虚引用对于对象而言完全是无感知的,有没有完全一样,但是对于虚引用的使用者而言,就像是待观察的对象的把脉线,可以通过它来观察对象是否已经被回收,从而进行相应的处理。
虚引用与软引用、弱引用的一个区别在于:虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。
@Test
public void testPhantomReference() throws InterruptedException {
//虚引用:功能,不会影响到对象的生命周期的,
// 但是能让程序员知道该对象什么时候被 回收了
ReferenceQueue<Object> referenceQueuee = new ReferenceQueue<>();
Object phantomObject = new Object();
PhantomReference phantomReference = new PhantomReference(phantomObject,referenceQueuee);
phantomObject = null;
System.out.println("phantomObject:" + phantomObject);//null
System.out.println("phantomReference" + referenceQueuee.poll());//null
System.gc();
Thread.sleep(2000);
System.out.println("referenceQueuee:" + referenceQueuee.poll());
}
测试的结果:
phantomObject:null
phantomReferencenull
referenceQueuee:java.lang.ref.PhantomReference@694f9431
5、补充
GC是需要2次扫描才回收对象,所以我们可以用finalize去救活丢失引用的对象
例如:
static App a;
@Override
protected void() throws Throwable{
super.finalize();
a=this;
}
6、总结
引用类型 | 被垃圾回收时间 | 用途 | 生命时间 |
---|---|---|---|
强引用 | 从来不会 | 对象的一般状态 | JVM停止运行时终止 |
软引用 | 在内存不足时 | 对象缓存 | 内存不足时终止 |
弱引用 | 在垃圾回收时 | 对象缓存 | GC运行后终止 |
虚引用 | Unknown | Unknown | Unknown |