本文介绍: 在虚拟机即将回收对象之前,如果发现对象存在引用,则会在回收对象后会将引用加入关联引用队列中。需要特别说明的是,并不是只有虚引用才能与引用队列关联,软引用和弱引用可以引用队列关联,只是说虚引用唯一作用就是感知对象垃圾回收时机。强引用指向对象是强可达的,而其他引用指向对象都是弱可达的。弱引用需要用 WeakReference 类来实现,它比软引用的生存期更短,对于只有弱引用的对象来说,只要垃圾回收机制运行,不管 JVM 的内存空间是否足够,总会回收该对象占用内存

JVM 系列吊打面试官:说一下 Java 的四种引用类型

image-20231205205234538

四种引种类型
1.强引用

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

2.软引用

软引用需要用 SoftReference 类来实现,对于只有软引用的对象来说,当系统内存足够时它不会被回收,当系统内存空间不足时它会被回收。软引用通常用在对内存敏感程序中。

3.弱引用

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

4.虚引用

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

我将它们的区别概括为 3 个维度

除了我们熟悉的四大引用,虚拟机内部设计一个 @hideFinalizerReference 引用,用于支持 Java Finalizer 机制,更多内容见 Finalizer 机制。

指针、引用和句柄什么区别?

引用、指针句柄具有指向对象地址的含义,可以将它们都简单理解一个内存地址。只有在具体的问题中,才需要区分它们的含义:

直接指针访问

image-20231205205712385

句柄访问

image-20231205205745494

引用使用方法

一节我们讨论如何将引用与引用队列的使用方法

使用引用对象

get() 和 clear() 最终是调用 native 方法我们在后文分析

SoftReference.java
// 已简化
public class SoftReference<T&gt; extends Reference<T&gt; {

    public SoftReference(T referent) {
        super(referent);
    }

    public SoftReference(T referent, ReferenceQueue<? super T&gt; q) {
        super(referent, q);
    }
}
WeakReference.java
public class WeakReference<T&gt; extends Reference<T&gt; {

    public WeakReference(T referent) {
        super(referent);
    }

    public WeakReference(T referent, ReferenceQueue<? super T&gt; q) {
        super(referent, q);
    }
}
PhantomReference.java
public class PhantomReference<T&gt; extends Reference<T&gt; {

    // 虚引用 get() 永远返回 null
    public T get() {
        return null;
    }

    // 虚引用必须管理引用队列,否则没有意义
    public PhantomReference(T referent, ReferenceQueue<? super T&gt; q) {
        super(referent, q);
    }
}
Reference.java
// 引用对象公共父类
public abstract class Reference<T> {

    // 虚拟机内部使用
    volatile T referent;

    // 关联引用队列
    final ReferenceQueue<? super T> queue;
 
    Reference(T referent) {
        this(referent, null);
    }

    Reference(T referent, ReferenceQueue<? super T> queue) {
        this.referent = referent;
        this.queue = queue;
    }

    // 获取引用指向的实际对象
    public T get() {
        // 调用 Native 方法
        return getReferent();
    }

    @FastNative
    private final native T getReferent();

    // 解除引用与实际对象的关联关系
    public void clear() {
        // 调用 Native 方法
        clearReferent();
    }

    @FastNative
    native void clearReferent();
    ...
}
引用队列使用模板

以下为 ReferenceQueue 的使用模板,主要分为 2 个阶段

示例程序
// 阶段 1:
// 创建对象
String strongRef = new String("abc");
// 1、创建引用队列
ReferenceQueue<String> referenceQueue = new ReferenceQueue<>();
// 2、创建引用对象,并关联引用队列
WeakReference<String> weakRef = new WeakReference<>(strongRef, referenceQueue);
System.out.println("weakRef 1:" + weakRef);
// 3、断开强引用
strongRef = null;

System.gc();

// 阶段 2:
// 延时 5000 是为了提高 "abc" 被回收的概率
view.postDelayed(new Runnable() {
    @Override
    public void run() {
        System.out.println(weakRef.get()); // 输出 null
        // 观察引用队列
        Reference<? extends String> ref = referenceQueue.poll();
        if (null != ref) {
            System.out.println("weakRef 2:" + ref);
            // 虽然可以获取到 Reference 对象,但无法获取到引用原本指向的对象
            System.out.println(ref.get()); // 输出 null
        }
    }
}, 5000);
程序输出
I/System.out: weakRef 1:java.lang.ref.WeakReference@3286da7
I/System.out: null
I/System.out: weakRef 2:java.lang.ref.WeakReference@3286da7
I/System.out: null

ReferenceQueue 中大部分 API 是面向 Java 虚拟机内部的,只有 ReferenceQueue#poll() 是面向开发者的。它是非阻塞 API,在队列有数据时返回队头的数据,而在队列为空时直接返回 null。

ReferenceQueue.java
public Reference<? extends T> poll() {
    synchronized (lock) {
        if (head == null)
            return null;

        return reallyPollLocked();
    }
}
工具类 Cleaner 使用模板

Cleaner 是虚引用的工具类,用于实现在对象被垃圾回收时额外执行一段清理逻辑本质上只是将虚引用和引用队列等代码做了简单封装而已。以下为 Cleaner 的使用模板

示例程序
// 1、创建对象
String strongRef = new String("abc");
// 2、创建清理逻辑
CleanerThunk thunk = new CleanerThunk();
// 3、创建 Cleaner 对象(本质上是一个虚引用)
Cleaner cleaner = Cleaner.create(strongRef, thunk);

private class CleanerThunk implements Runnable {
    @Override
    public void run() {
        // 清理逻辑
    }
}
Cleaner.java
// Cleaner 只不过是虚引用的工具类而已
public class Cleaner extends PhantomReference<Object> {
    ...
}

jvm相关知识点持续更新中!喜欢的话请点赞收藏关注哦!

原文地址:https://blog.csdn.net/qq_24428851/article/details/134817495

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任

如若转载,请注明出处:http://www.7code.cn/show_46536.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱suwngjj01@126.com进行投诉反馈,一经查实,立即删除

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注