四种引用——强引用、弱引用、软引用和虚引用

Java执行GC判断对象是否存活有两种方式其中一种是引用计数

引用计数:Java堆中每一个对象都有一个引用计数属性,引用每新增1次计数加1,引用每释放1次计数减1。

JDK 1.2以前的版本中,若一个对象不被任何变量引用,那么程序就无法再使用这个对象。也就是说,只有对象处于(reachable)可达状态,程序才能使用它。

JDK 1.2版本开始,对象的引用被划分为4种级别,从而使程序能更加灵活地控制对象的生命周期。这4种级别由高到低依次为:强引用、软引用、弱引用和虚引用

强引用 (StrongReference)

在 Java 中最常见的就是强引用,把一个对象赋给一个引用变量,这个引用变量就是一个强引
。当一个对象被强引用变量引用时,它处于可达状态,它是不可能被垃圾回收机制回收的,即
使该对象以后永远都不会被用到 JVM 也不会回收。因此强引用是造成 Java 内存泄漏的主要原因之
一。

 Object strongReference = new Object();

当内存空间不足时,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足的问题。


在一个方法的内部有一个强引用,这个引用保存在Java中,而真正的引用内容(Object)保存在Java中。当这个方法运行完成后,就会退出方法栈,则引用对象的引用数为0,这个对象会被回收。

public void test() {
   Object strongReference = new Object();
   // 省略其他操作
}

显式地设置strongReference对象为null,或让其超出对象的生命周期范围,则GC认为该对象不存在引用,这时就可以回收这个对象。具体什么时候收集这要取决于GC算法。

 strongReference = null;

软引用 (SoftReference)

软引用需要用 SoftReference 类来实现,对于只有软引用的对象来说,当系统内存足够时它
不会被回收,当系统内存空间不足时它会被回收

软引用通常用在对内存敏感的程序中。

// 强引用
String strongReference = new String("abc");
// 软引用
String str = new String("abc");
SoftReference<String> softReference = new SoftReference<String>(str);

GC线程会在虚拟机抛出OutOfMemoryError之前回收软引用对象,而且虚拟机会尽可能优先回收长时间闲置不用的软引用对象。对那些刚构建的或刚使用过的“较新的”软对象会被虚拟机尽可能保留,这就是引入引用队列ReferenceQueue的原因。

弱引用(WeakReference)

弱引用需要用 WeakReference 类来实现,它比软引用的生存期更短,对于只有弱引用的对象
来说,只要垃圾回收机制一运行,不管 JVM 的内存空间是否足够,总会回收该对象占用的内存。

String str = new String("abc");
WeakReference<String> weakReference = new WeakReference<>(str);

如果一个对象是偶尔(很少)的使用,并且希望在使用时随时就能获取到,但又不想影响此对象的垃圾收集,那么你应该用Weak Reference来记住此对象。

虚引用(PhantomReference)

虚引用需要 PhantomReference 类来实现,它不能单独使用,必须和引用队列联合使用。
引用的主要作用是跟踪对象被垃圾回收的状态。

虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。

String str = new String("abc");
ReferenceQueue queue = new ReferenceQueue();
// 创建虚引用,要求必须与一个引用队列关联
PhantomReference pr = new PhantomReference(str, queue);

ReferenceQueue

引用队列,在检测到适当的可到达性更改后,垃圾回收器将已注册的引用对象添加到该队列中。

SoftReference、WeakReference的构造方法都有ReferenceQueue的重载

ReferenceQueue主要是用于监听Reference所指向的对象是否已经被垃圾回收

当大量使用Reference时,虽然Reference指向的对象可能被回收了,但Reference本身也是个对象,所以也需要回收。这时就需要使用ReferenceQueue了。

当SoftReference或WeakReference的get()加入ReferenceQueue或get()返回null时,仅是表明其指向的对象已经进入垃圾回收流程,此时对象不一定已经被垃圾回收。当PhantomReference加入ReferenceQueue时,则表明对象已经被回收。

ReferenceQueue<String> referenceQueue = new ReferenceQueue<>();
String str = new String("abc");
SoftReference<String> softReference = new SoftReference<>(str, referenceQueue);

str = null;
// Notify GC
System.gc();

System.out.println(softReference.get()); // abc

Reference<? extends String> reference = referenceQueue.poll();
System.out.println(reference); //null

总结

Java中4种引用的级别和强度由高到低依次为:强引用 -> 软引用 -> 弱引用 -> 虚引用

当垃圾回收器回收时,某些对象会被回收,某些不会被回收。垃圾回收器会从根对象Object来标记存活的对象,然后将某些不可达的对象和一些引用的对象进行回收。

引用类型 被垃圾回收时间 用途 生存时间
强引用 从来不会 对象的一般状态 JVM停止运行时终止
软引用 当内存不足时 对象缓存 内存不足时终止
弱引用 正常垃圾回收时 对象缓存 垃圾回收后终止
虚引用 正常垃圾回收时 跟踪对象的垃圾回收 垃圾回收后终止

参考文献

https://blog.csdn.net/baidu_22254181/article/details/82555485

posted @ 2022-02-23 13:22  城南孔乙己  阅读(165)  评论(0)    收藏  举报