Java 四种引用总结(强引用、软引用、弱引用、虚引用)

1、强引用(StrongReference)

也就是我们new出来的对象当内存不足的时候,JVM宁可出现OutOfMemory错误停止,也需要进行保存,并且不会将此空间回收(永远不会被GC回收);

例如:

Object obj = new Object();

2、软引用(SoftReference)

内存不足时回收,存放一些重要性不是很强又不能随便让清除的对象,比如图片切换到后台不需要马上显示了.

软引用可用来实现内存敏感的高速缓存。

Object obj = new Object();  //强引用
ReferenceQueue 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 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 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

你可能感兴趣的:(Java 四种引用总结(强引用、软引用、弱引用、虚引用))