Java 面试题:强引用、软引用、弱引用、幻象引用有什么区别?

在 Java 中,理解不同类型引用的区别对于掌握内存管理和垃圾回收机制是至关重要的。强引用、软引用、弱引用和幻象引用分别提供了不同的对象引用强度,使开发者能够精细控制对象的生命周期和内存使用情况。

强引用(Strong Reference)是 Java 中最常见的引用类型。当一个对象被一个强引用所引用时,垃圾回收器永远不会回收该对象,即使内存不足,JVM 也会抛出 OutOfMemoryError,而不会回收此对象。

软引用(Soft Reference)在内存不足时会被垃圾回收器回收。它非常适合用来实现缓存,当内存充足时可以保留缓存数据,而在内存不足时则允许垃圾回收器回收这些缓存数据以释放内存。

弱引用(Weak Reference)在垃圾回收器运行时,只要发现一个对象仅被弱引用引用,就会立即回收该对象。弱引用通常用于实现引用敏感的映射,比如 WeakHashMap,其键值对在键不再被使用时可以自动被回收。

幻象引用(Phantom Reference)在任何时候都可能被垃圾回收器回收。它主要用于跟踪对象被垃圾回收的时间,与引用队列一起使用,能够在对象被回收后进行一些清理操作,但无法通过幻象引用访问对象本身。

通过深入理解这些引用类型的区别,你将更好地掌握 Java 的内存管理机制,写出更加高效和健壮的应用程序。


文章目录

      • 1、面试问题
      • 2、问题分析
      • 3、典型回答
      • 4、问题深入
        • 4.1、解释不同引用类型在垃圾收集中的具体行为
        • 4.2、讨论软引用和弱引用在缓存实现中的应用
        • 4.3、解释幻象引用和引用队列的关系及应用场景
        • 4.4、提供示例代码展示不同引用类型的使用
        • 4.5、讨论不同引用类型的性能影响和使用注意事项
        • 4.6、解释引用类型在Java中的实现机制


1、面试问题

今天的面试问题:Java 的强引用、软引用、弱引用、幻象引用有什么区别?


2、问题分析

这个问题主要考察以下几个关键点:

  1. 引用类型的基本概念:了解Java中不同类型的引用及其定义。
  2. 垃圾收集器的行为:理解不同引用类型对垃圾收集的影响。
  3. 引用类型的应用场景:掌握每种引用类型的典型应用场景。
  4. 引用的实现机制:知道引用类型在Java中的实现方式及其相关类。

这个问题不仅考察基础知识,还涉及Java内存管理和垃圾收集机制,是评估Java开发者技能的一个重要方面。


3、典型回答

Java 中有四种引用类型:强引用、软引用、弱引用和幻象引用。它们在可达性和垃圾收集方面有不同的行为和应用场景。

强引用(Strong Reference)

  • 定义:最常见的普通对象引用。
  • 特点:只要有强引用指向一个对象,垃圾收集器就不会回收该对象。
  • 示例:
Object obj = new Object(); // 强引用
  • 应用场景:适用于需要长期持有的对象,如大部分的普通对象。

软引用(Soft Reference)

  • 定义:一种相对强引用弱化的引用,可以让对象豁免一些垃圾收集。
  • 特点:只有当JVM认为内存不足时,才会回收软引用指向的对象。确保在抛出OutOfMemoryError之前,清理软引用指向的对象。
  • 示例:
SoftReference<Object> softRef = new SoftReference<>(new Object());
  • 应用场景:适用于实现内存敏感的缓存,如缓存数据在内存不足时会被回收。

弱引用(Weak Reference)

  • 定义:更弱的引用类型,不能使对象豁免垃圾收集。
  • 特点:只要垃圾收集器发现只有弱引用指向对象时,就会回收该对象。
  • 示例:
WeakReference<Object> weakRef = new WeakReference<>(new Object());
  • 应用场景:适用于维护非强制性的映射关系,如WeakHashMap中的键。

幻象引用(Phantom Reference)

  • 定义:最弱的引用类型,无法通过幻象引用访问对象。
  • 特点:用于跟踪对象被垃圾收集器回收的状态,必须与引用队列(ReferenceQueue)一起使用。
  • 示例:
PhantomReference<Object> phantomRef = new PhantomReference<>(new Object(), new ReferenceQueue<>());
  • 应用场景:用于实现对象被回收时的清理机制,如在对象被finalize后执行某些操作。

4、问题深入

4.1、解释不同引用类型在垃圾收集中的具体行为

强引用(Strong Reference)

  • 定义:最常见的普通对象引用。
  • 垃圾收集行为:只要有强引用指向一个对象,垃圾收集器就不会回收该对象。即使内存不足,垃圾收集器也不会回收有强引用的对象。
  • 应用场景:适用于需要长期持有的对象,如大部分的普通对象。

示例:

Object strongRef = new Object(); // 强引用

软引用(Soft Reference)

  • 定义:一种相对强引用弱化的引用。
  • 垃圾收集行为:只有当JVM认为内存不足时,才会回收软引用指向的对象。确保在抛出OutOfMemoryError之前,清理软引用指向的对象。
  • 应用场景:适用于实现内存敏感的缓存,如缓存数据在内存不足时会被回收。

示例:

SoftReference<Object> softRef = new SoftReference<>(new Object());

弱引用(Weak Reference)

  • 定义:更弱的引用类型,不能使对象豁免垃圾收集。
  • 垃圾收集行为:只要垃圾收集器发现只有弱引用指向对象时,就会回收该对象。
  • 应用场景:适用于维护非强制性的映射关系,如WeakHashMap中的键。

示例:

WeakReference<Object> weakRef = new WeakReference<>(new Object());

幻象引用(Phantom Reference)

  • 定义:最弱的引用类型,无法通过幻象引用访问对象。
  • 垃圾收集行为:用于跟踪对象被垃圾收集器回收的状态,必须与引用队列(ReferenceQueue)一起使用。
  • 应用场景:用于实现对象被回收时的清理机制,如在对象被finalize后执行某些操作。

示例:

PhantomReference<Object> phantomRef = new PhantomReference<>(new Object(), new ReferenceQueue<>());
4.2、讨论软引用和弱引用在缓存实现中的应用

软引用缓存

  • 特点:在内存充足时保留缓存对象,当内存不足时回收缓存,确保内存不被耗尽。
  • 应用场景:适用于缓存不需要立即使用的数据,例如图片缓存,能够在内存不足时自动回收。

示例:

import java.lang.ref.SoftReference;
import java.util.HashMap;
import java.util.Map;public class SoftReferenceCache {private Map<String, SoftReference<Object>> cache = new HashMap<>();public void put(String key, Object value) {cache.put(key, new SoftReference<>(value));}public Object get(String key) {SoftReference<Object> ref = cache.get(key);if (ref != null) {return ref.get();}return null;}
}

弱引用缓存

  • 特点:在对象不再被其他强引用引用时回收缓存对象,避免内存泄漏。
  • 应用场景:适用于缓存需要动态清理的对象,例如维护非强制性的映射关系。

示例:

import java.lang.ref.WeakReference;
import java.util.HashMap;
import java.util.Map;public class WeakReferenceCache {private Map<String, WeakReference<Object>> cache = new HashMap<>();public void put(String key, Object value) {cache.put(key, new WeakReference<>(value));}public Object get(String key) {WeakReference<Object> ref = cache.get(key);if (ref != null) {return ref.get();}return null;}
}
4.3、解释幻象引用和引用队列的关系及应用场景

幻象引用和引用队列

  • 关系:幻象引用必须与引用队列一起使用。引用队列用于跟踪对象的回收状态。当对象被回收时,幻象引用会被加入到引用队列中,可以通过队列处理回收后的清理操作。
  • 应用场景:用于实现对象被回收时的清理机制,如在对象被finalize后执行某些操作。

示例:

import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;public class PhantomReferenceExample {public static void main(String[] args) throws InterruptedException {Object obj = new Object();ReferenceQueue<Object> refQueue = new ReferenceQueue<>();PhantomReference<Object> phantomRef = new PhantomReference<>(obj, refQueue);// 清除强引用obj = null;// 触发垃圾收集System.gc();// 检查引用队列,处理回收后的清理操作Reference<? extends Object> ref = refQueue.remove();if (ref == phantomRef) {System.out.println("Object has been garbage collected");}}
}
4.4、提供示例代码展示不同引用类型的使用

强引用示例

Object strongRef = new Object();

软引用示例

import java.lang.ref.SoftReference;public class SoftReferenceExample {public static void main(String[] args) {SoftReference<Object> softRef = new SoftReference<>(new Object());Object obj = softRef.get();if (obj == null) {obj = new Object();softRef = new SoftReference<>(obj);}}
}

弱引用示例

import java.lang.ref.WeakReference;public class WeakReferenceExample {public static void main(String[] args) {WeakReference<Object> weakRef = new WeakReference<>(new Object());Object obj = weakRef.get();if (obj == null) {obj = new Object();weakRef = new WeakReference<>(obj);}}
}

幻象引用示例

import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;public class PhantomReferenceExample {public static void main(String[] args) throws InterruptedException {Object obj = new Object();ReferenceQueue<Object> refQueue = new ReferenceQueue<>();PhantomReference<Object> phantomRef = new PhantomReference<>(obj, refQueue);// 清除强引用obj = null;// 触发垃圾收集System.gc();// 检查引用队列,处理回收后的清理操作Reference<? extends Object> ref = refQueue.remove();if (ref == phantomRef) {System.out.println("Object has been garbage collected");}}
}
4.5、讨论不同引用类型的性能影响和使用注意事项

强引用

  • 性能影响:最常用,性能最好,不会被垃圾收集器回收。
  • 注意事项:避免内存泄漏,确保在不再需要对象时将其引用置为null。

软引用

  • 性能影响:在内存不足时会回收软引用对象,可能影响性能。
  • 注意事项:适用于实现内存敏感的缓存,但需要注意内存管理,避免频繁回收。

弱引用

  • 性能影响:弱引用对象在垃圾收集时会被回收,频繁使用可能导致性能问题。
  • 注意事项:适用于非强制性关系的维护,避免内存泄漏,但可能导致频繁回收影响性能。

幻象引用

  • 性能影响:用于跟踪对象回收状态,不会影响对象的生命周期。
  • 注意事项:适合高级内存管理和清理操作,但实现复杂,需要与引用队列一起使用。
4.6、解释引用类型在Java中的实现机制

Java中 的引用类型

Java通过java.lang.ref包提供了引用类型的实现,包括SoftReferenceWeakReferencePhantomReference类。这些类继承自Reference类,提供了对不同引用类型的支持。

SoftReference类:

public class SoftReference<T> extends Reference<T> {// 具体实现省略
}

WeakReference类:

public class WeakReference<T> extends Reference<T> {// 具体实现省略
}

PhantomReference类:

public class PhantomReference<T> extends Reference<T> {// 具体实现省略
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/32247.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

rapidjson使用中crash问题分析

问题 在使用rapidjson时&#xff0c;使用Document的Parse方法解析json字符串&#xff0c;程序crash。 分析 可以参考https://github.com/Tencent/rapidjson/issues/1269&#xff0c;由于rapidjson的内存分配器默认认为内存分配成功&#xff0c;没有对分配后做判空判断&#…

35.简易远程数据框架的实现

上一个内容&#xff1a;34.构建核心注入代码 34.构建核心注入代码它的调用LoadLibrary函数的代码写到游戏进程中之后无法调用&#xff0c;动态链接库的路径是一个内存地址&#xff0c;写到游戏进程中只把内存地址写过去了&#xff0c;内存地址里的内容没写过去&#xff0c;导致…

Netty ByteBuf 使用详解

文章目录 1.概述2. ByteBuf 分类3. 代码实例3.1 常用方法3.1.1 创建ByteBuf3.1.2 写入字节3.1.3 扩容3.1.2.1 扩容实例3.1.2.2 扩容计算新容量代码 3.1.4 读取字节3.1.5 标记回退3.1.6 slice3.1.7 duplicate3.1.8 CompositeByteBuf3.1.9 retain & release3.1.9.1 retain &a…

Jmeter5.X性能测试【完整版】

目录 一、Http基础协议和解析 1、浏览器的B/S架构和C/S架构 &#xff08;1&#xff09;CS架构 &#xff08;2&#xff09;BS架构 &#xff08;3&#xff09;URL理解 2、Http超文本传输协议 &#xff08;1&#xff09;含义 # 协议 # json协议 # xml协议 &#xff08;…

分布式锁(Redission)

分布式锁&#xff1a; 使用场景&#xff1a; 通常对于一些使用率高的服务&#xff0c;我们会进行多次部署&#xff0c;可能会部署在不同的服务器上&#xff0c;但是他们获取和操作的数据仍然是同一份。为了保证服务的强一致性&#xff0c;我们需要对线程进行加锁&#xff0c;…

WRONGPASS invalid username-password pair or user is disabled

连接redis客户端的时候报错&#xff1a;WRONGPASS invalid username-password pair or user is disabled 当前redis版本为7.2.4 一、问题分析 默认情况下&#xff0c;Redis 7.0 使用默认用户名 default 和空密码进行身份验证。如果未设置 requirepass&#xff0c;则默认用户名…

C++实现自动生成c++类中的属性的get和set方法

目录 应用场景 运行准备 代码展示 结果显示 应用场景 当我们在编写类的属性时&#xff0c;需要对该属性进行封装&#xff0c;需要一系列的get和set的方法。例如下面是天气类的成员属性。可以看到属性很多&#xff0c;而写get和set都是一些固定的操作&#xff0c;因此可以直…

任务4.8.4 利用Spark SQL实现分组排行榜

文章目录 1. 任务说明2. 解决思路3. 准备成绩文件4. 采用交互式实现5. 采用Spark项目实战概述&#xff1a;使用Spark SQL实现分组排行榜任务背景任务目标技术选型实现步骤1. 准备数据2. 数据上传至HDFS3. 启动Spark Shell或创建Spark项目4. 读取数据5. 数据转换6. 创建临时视图…

哈夫曼树及其应用

目录 一、哈夫曼树 1.1基本概念 1.2构造方法 1.3构造算法的实现 二、哈夫曼树的应用 2.1哈夫曼编码 2.2文件的编码和解码 2.2.1编码 2.2.2解码 一、哈夫曼树 1.1基本概念 哈夫曼树又称为最优树&#xff0c;是一类带权路径长度最短的树。 最优二叉树&#xff1a;带权…

Android内核编译

前言 本文描述使用Ubuntu 编译Android内核刷入pixel4一些心得和流程。 PC信息&#xff1a; ./o- jackjackyyyyy- -yyyyyy OS: Ubuntu 22.04 jammy:////-yyyyyyo Kernel: x86_64 Linux 6.5.0-35-generic. .://-.sss/ Uptime: 1d 5h 4m.:o: //:--:/- …

Java 时间日期类API

Java 关于日期时间API 文章目录 Java 关于日期时间APIJDK8之前System类的方法java.util.DateJava.sql.Datejava.text.SimpleDateFormatjava.util.Calendar(日历) JDK8.0&#xff0c;新日期时间APIjava.timeInstant 瞬时与传统日期处理的转换 Java中关于日期时间API分为JDK8之前…

C++入门 vector部分模拟实现

目录 vector大致框架 vector常见接口模拟实现 begin迭代器 & end迭代器 capacity( ) & size( ) reserve operator[ ] push_back( ) & pop_back( ) sort vector大致框架 vector的内部的成员变量大概有三部分构成&#xff1a; namespace bit {template<c…

【Java算法】滑动窗口 上

&#x1f525;个人主页&#xff1a; 中草药 &#x1f525;专栏&#xff1a;【算法工作坊】算法实战揭秘 &#x1f456;一. 长度最小的子数组 题目链接&#xff1a;209.长度最小的子数组 算法原理 滑动窗口 滑动窗口算法常用于处理数组/字符串等序列问题&#xff0c;通过定义一…

Java赋值运算符

Java赋值运算符分为以下&#xff1a; 符号 作用 说明 赋值 int a 10,把10赋值给变量a 加后赋值 ab,将ab的值赋值给变量a - 减后赋值 a-b,将a-b的值赋值给变量a* 乘后赋值 a*b,将a*b的值赋值给变量a / 除后赋值 a/b,将a/b的值赋值给变量a % 取余赋值 a%b,将a%b的值赋值给变量…

力扣84.柱状图中最大的矩形

力扣84.柱状图中最大的矩形 初始化pre_max 为-1 存距离最近的小于h[i]的元素下标 初始化suf_max 为 n 存距离最近的小于h[i]的元素下标 class Solution {public:int largestRectangleArea(vector<int>& heights) {int n heights.size();//分别初始化-1 和 nvect…

C#循环语句总结

前言 正所谓磨刀不误砍柴工&#xff0c;C#上位机软件开发能力的提升离不开对C#语法的精通&#xff0c;本文接着讲解C#语法知识中的循环语句&#xff0c;在C#程序开发中我们经常会用到各种循环语句&#xff0c;常见的有for循环、while循环&#xff0c;本文就是对C#中用到的各种…

贪心算法—

贪心算法是一种在每一步选择中都采取在当前状态下最好或最优&#xff08;即最有利&#xff09;的选择&#xff0c;从而希望导致结果是全局最好或最优的算法。这种算法并不总是能找到全局最优解&#xff0c;但在某些问题上能提供足够好的解决方案。贪心算法的关键特性包括&#…

drogon orm分页问题,req->getJsonObject()为空会导致Segmentation fault

2024年6月22日17:14:12 req->getJsonObject()获取json数据的时候&#xff0c;如果没有提前判断 if (req->getJsonObject() nullptr){throw std::invalid_argument("参数json不能为空");}auto jsonPtr req->getJsonObject();官方文档&#xff1a;https://…

JR-8000系列机架式多路4K超高清光端机

集中式 4K超高清光传输设备 1 产品特性 ⚫ 支持高达 8 通道 SMPTE 全格式 SDI 信号输入 ⚫ 发送端带有 LOOPOUT 环出端口&#xff0c;具备消抖动功能&#xff0c;可作为信号调理或级联信号源使用 ⚫ 接收端支持双输出端口 ⚫ 支持传输速率&#xff1a;143Mbps-11.88Gbps ⚫…

Intel太无耻,跟着玩数字游戏还揭台积电的老底,工艺都是假的

在台积电的3纳米逐渐获得芯片企业认可的情况下&#xff0c;近日Intel却再次指出台积电的3纳米工艺并非真正的3纳米&#xff0c;与Intel的7纳米工艺差不多&#xff0c;这显示出Intel在芯片工艺研发方面日益落后的情况下确实有点慌了。 Intel指出它的7纳米工艺的晶体管密度达到1.…