Java中四大引用

强引用(StrongReference)

概念介绍:

在此说明一下,StrongReference只是对强引用的一个称呼,但是强引用没有对应的实体类。使用强引用的对象就算是内存出现OutOfMemory(内存溢出)的异常也不会回收。也就是说该对象永远不会被垃圾回收器回收,不论内存是否充足。平常代码中用的最多的就是强引用。

Demo验证:

static Object object = new Object();
.......................................
/**
 * 强引用
 */
public static void testStrongReference(){
    Object obj = object;
    object = null;
    System.gc();
    System.out.print("after system.gc-strongReference---obj = " + obj);
}

输出结果

exclude patterns:
after system.gc-strongReference---obj = java.lang.Object@4a574795Class

Process finished with exit code 0

在该demo中obj就是使用的默认的强引用,虽然obj所指向的对象被置为null,但gc不会回收该强引用对象

软引用(SoftReference)

概念介绍:

软引用有对应的实体列为SoftReference,使用软引用引用的对象只有在程序发生oom异常前才会回收,也就是说如果内存充足永远不会被回收,只有在内存不足时才会回收,很好的避免oom,非常适合做缓存。

Demo验证

/**
 * 软引用
 */
public static void testSoftReference() {
    SoftReference<Object> obj = new SoftReference<>(object);
    object = null;
    System.gc();
    System.out.print("after system.gc---softReference = " + obj);
}

输出结果

exclude patterns:
after system.gc---softReference = java.lang.ref.SoftReference@4a574795
Process finished with exit code 0

由控制台的输出结果可以看到,虽然软引用引用的对象被清空,但是由于内存充足,就算是执行了gc也不会被回收。

小结一下:由以上的demo可以看到软引用和强引用在内存充足的情况下是一样的都不会被回收,只有在内存不足时软引用才会被及时回收避免oom异常,而强引用却很霸道坚决不回收。这样分析来看,使用软引用可以很好的避免oom异常,适合做一些缓存工作

弱引用(WeakReference)

概念介绍:

弱引用也是用来描述非必需对象的,当JVM进行垃圾回收时,无论内存是否充足,都会回收被弱引用关联的对象。不过由于垃圾回收器是一个优先级较低的线程,所以并不一定能迅速发现弱引用对象。参考以下demo进行理解。

Demo验证:

第一种情况,如果把弱引用所引用的对象置为null,但不进行gc,弱引用的get到的会是null吗?

/**
 * 弱引用
 */
public static void testWeakReference() {
    WeakReference<Object> weakReference = new WeakReference<Object>(object);
    WeakReference<Object> weakReferenceStr = new WeakReference<Object>(str);
    object = null;
    str = null;
//  System.gc();
    System.out.println("after system.gc---weakReference = " + weakReference.get());
    System.out.print("after system.gc---weakReferenceStr = " + weakReferenceStr.get());
}

输出结果为:

exclude patterns:
after system.gc---weakReference = java.lang.Object@4a574795
after system.gc---weakReferenceStr = strTest

Process finished with exit code 0

可以看到如果不进行gc,弱引用对象在内存充足的情况下是不会回收的,不论所引用的对象是否为null
第二种情况,不对所引用的对象置为null,直接进行gc

/**
 * 弱引用
 */
public static void testWeakReference() {
    WeakReference<Object> weakReference = new WeakReference<Object>(object);
    WeakReference<Object> weakReferenceStr = new WeakReference<Object>(str);
//    object = null;
//    str = null;
    System.gc();
    System.out.println("after system.gc---weakReference = " + weakReference.get());
    System.out.print("after system.gc---weakReferenceStr = " + weakReferenceStr.get());
}

输出结果为

exclude patterns:
after system.gc---weakReference = java.lang.Object@4a574795
after system.gc---weakReferenceStr = strTest

Process finished with exit code 0

可以看到,此时所引用的对象不为null,所以在进行gc回收时不会回收弱引用对象。由此可以看出弱引用跟对象的生命周期有关,在对象不为null时,垃圾回收器不会回收弱引用

第三种情况

/**
 * 弱引用
 */
public static void testWeakReference() {
    WeakReference<Object> weakReference = new WeakReference<Object>(object);
    WeakReference<Object> weakReferenceStr = new WeakReference<Object>(str);
    object = null;
    str = null;
    System.gc();
    System.out.println("after system.gc---weakReference = " + weakReference.get());
    System.out.print("after system.gc---weakReferenceStr = " + weakReferenceStr.get());
}

输出结果

exclude patterns:
after system.gc---weakReference = null
after system.gc---weakReferenceStr = strTest

Process finished with exit code 0

在这里说明一下str对象的定义方式

static String str = "strTest";

可以看到,使用new的方式构造的对象在置为null后,所对应的弱引用被回收,但是使用赋值的方式所获取的对象置为null后,所对应的弱引用没有被回收,这是为什么呢?这里暂且TODO该问题,把四大引用总结完后再研究

这是因为gc不清理常量池里的垃圾,所以所引用的内容不为null

综合以上三种情况,可以对弱引用下一个结论了,弱应用与对象的生命周期有关。在进行垃圾回收时,如果所引用的对象为null,则不论内存是否充足都会被回收,否则不会被回收。当然当内存不足时会直接被回收。

虚引用(PhantonReference)

概念介绍

虚引用对应的实体类为PhantonReference。虚引用不论所引用的对象是不是null不论内存空间是否充足,都会被垃圾回收器回收

Demo验证

/**
 * 虚引用
 */
public static void testPhantonReference() {
    ReferenceQueue<Object> referenceQueue = new ReferenceQueue<>();
    PhantomReference<Object> phantomReference = new PhantomReference<>(object, referenceQueue);
    PhantomReference<Object> phantomReferenceStr = new PhantomReference<>(str, referenceQueue);
//  object = null;
//  str = null;
    System.gc();
    System.out.println("after system.gc---phantomReference = " + phantomReference.get());
    System.out.print("after system.gc---phantomReferenceStr = " + phantomReferenceStr.get());
}

输出结果如下
 

exclude patterns:
after system.gc---phantomReference = null
after system.gc---phantomReferenceStr = null
Process finished with exit code 0


可以看到,只要进行垃圾回收,虚引用就会被回收

总结:

对于垃圾回收器回收的顺序为

虚引用---弱引用----软引用---强引用。

多使用软引用做缓存可以很好地避免oom.

参考:https://blog.csdn.net/zrf1335348191/article/details/53138648

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值