在 Java 中,理解不同类型引用的区别对于掌握内存管理和垃圾回收机制是至关重要的。强引用、软引用、弱引用和幻象引用分别提供了不同的对象引用强度,使开发者能够精细控制对象的生命周期和内存使用情况。
强引用(Strong Reference)是 Java 中最常见的引用类型。当一个对象被一个强引用所引用时,垃圾回收器永远不会回收该对象,即使内存不足,JVM 也会抛出 OutOfMemoryError,而不会回收此对象。
软引用(Soft Reference)在内存不足时会被垃圾回收器回收。它非常适合用来实现缓存,当内存充足时可以保留缓存数据,而在内存不足时则允许垃圾回收器回收这些缓存数据以释放内存。
弱引用(Weak Reference)在垃圾回收器运行时,只要发现一个对象仅被弱引用引用,就会立即回收该对象。弱引用通常用于实现引用敏感的映射,比如 WeakHashMap,其键值对在键不再被使用时可以自动被回收。
幻象引用(Phantom Reference)在任何时候都可能被垃圾回收器回收。它主要用于跟踪对象被垃圾回收的时间,与引用队列一起使用,能够在对象被回收后进行一些清理操作,但无法通过幻象引用访问对象本身。
通过深入理解这些引用类型的区别,你将更好地掌握 Java 的内存管理机制,写出更加高效和健壮的应用程序。
今天的面试问题:Java 的强引用、软引用、弱引用、幻象引用有什么区别?
这个问题主要考察以下几个关键点:
这个问题不仅考察基础知识,还涉及Java内存管理和垃圾收集机制,是评估Java开发者技能的一个重要方面。
Java 中有四种引用类型:强引用、软引用、弱引用和幻象引用。它们在可达性和垃圾收集方面有不同的行为和应用场景。
强引用(Strong Reference)
Object obj = new Object(); // 强引用
软引用(Soft Reference)
OutOfMemoryError之前,清理软引用指向的对象。SoftReference<Object> softRef = new SoftReference<>(new Object());
弱引用(Weak Reference)
WeakReference<Object> weakRef = new WeakReference<>(new Object());
幻象引用(Phantom Reference)
PhantomReference<Object> phantomRef = new PhantomReference<>(new Object(), new ReferenceQueue<>());
强引用(Strong Reference)
示例:
Object strongRef = new Object(); // 强引用
软引用(Soft Reference)
OutOfMemoryError之前,清理软引用指向的对象。示例:
SoftReference<Object> softRef = new SoftReference<>(new Object());
弱引用(Weak Reference)
示例:
WeakReference<Object> weakRef = new WeakReference<>(new Object());
幻象引用(Phantom Reference)
示例:
PhantomReference<Object> phantomRef = new PhantomReference<>(new Object(), new ReferenceQueue<>());
软引用缓存
示例:
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;
}
}
幻象引用和引用队列
示例:
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");
}
}
}
强引用示例
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");
}
}
}
强引用
软引用
弱引用
幻象引用
Java中 的引用类型
Java通过java.lang.ref包提供了引用类型的实现,包括SoftReference、WeakReference和PhantomReference类。这些类继承自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> {
// 具体实现省略
}