Java四种引用--《深入理解Java虚拟机》学习笔记及个人理解(四)

Java四种引用--《深入理解Java虚拟机》学习笔记及个人理解(四)

书上P65.

StrongReference(强引用)

类似Object obj = new Object() 这类的引用 就是强引用.

SoftReference(软引用)

用来描述一些有用但并非必需的对象. 在系统将要发生内存溢出异常前, 将会把这些对象列进回收范围之中, 进行二次回收.(如果这次回收了之后, 内存还是不足, 那么就会抛出内存溢出异常)

执行下面这段代码的时候, 设置一下虚拟机参数-Xms2m -Xmx2m. 设置最大堆内存. 不然程序要运行很久才能堆内存溢出....

import java.lang.ref.SoftReference;
import java.util.ArrayList;
import java.util.List; /**
* VM Options: -Xms2m -Xmx2m
*/
public class Main {
public static void main(String[] args) {
Dog<String> dog = new Dog<>(new String("1234321"));
System.out.println("内存溢出之前: " + dog.get()); // 1234321 try {
for (List<Object> arrayList = new ArrayList<>(); ; )
arrayList.add(new Object());
} catch (Throwable e) {
System.out.println("内存溢出时:" + dog.get()); // null
}
} static class Dog<V> extends SoftReference<Object> {
public Dog(V value) {
super(value);
}
}
}

输出结果如下:

Java四种引用--《深入理解Java虚拟机》学习笔记及个人理解(四)

WeakReference(弱引用)

弱引用也是用来描述非必需对象的, 强度比软引用还要弱. 被弱引用关联的对象只能生存到下一次垃圾收集器发生之前. 当垃圾收集器工作时, 无论当前内存是否足够, 都会被回收掉. (这里指的是"只有弱引用"的情况, 如果同时还被其他强引用关联了, GC Root可达, 那么当然不会被回收了.)

import java.lang.ref.WeakReference;

public class Main {
public static void main(String[] args) {
Integer a = new Integer(111);
String b = new String("222");
Node wrc = new Node(a, b);
System.out.println("gc之前: " + wrc.get());
System.out.println(wrc.value);
a = null;
System.gc();
System.out.println("gc之后: " + wrc.get());
System.out.println(wrc.value); }
} class Node<K, V> extends WeakReference<Object> {
V value; public Node(K key, V value) {
super(key);
this.value = value;
}
}

Java四种引用--《深入理解Java虚拟机》学习笔记及个人理解(四)

PhantomReference(虚引用, 幽灵引用, 幻影引用)

这是最弱的引用. 这个不会影响到对象的生存时间. 也无法通过ge()来获取到对象实例(永远是null).

不过可以用虚引用让一个对象被收集器回收时执行一段通知.

本段例子来自于https://www.baeldung.com/java-phantom-reference

/**
* https://www.baeldung.com/java-phantom-reference
*/
public class PhantomReferenceTest {
public static void main(String[] args) {
ReferenceQueue<Object> referenceQueue = new ReferenceQueue<>();
List<LargeObjectFinalizer> references = new ArrayList<>();
List<Object> largeObjects = new ArrayList<>(); for (int i = 0; i < 3; ++i) {
Object largeObject = new Object();
largeObjects.add(largeObject);
references.add(new LargeObjectFinalizer(largeObject, referenceQueue));
} for (PhantomReference<Object> reference : references) {
System.out.println(reference.isEnqueued());
} largeObjects = null;
System.gc(); for (PhantomReference<Object> reference : references) {
System.out.println(reference.isEnqueued());
} for (Reference<?> referenceFromQueue; (referenceFromQueue = referenceQueue.poll()) != null; referenceFromQueue.clear()) {
((LargeObjectFinalizer) referenceFromQueue).finalizeResources();
}
} static class LargeObjectFinalizer extends PhantomReference<Object> { LargeObjectFinalizer(Object referent, ReferenceQueue<? super Object> q) {
super(referent, q);
} void finalizeResources() {
System.out.println("clearing ...");
}
}
}

输出结果如下:

Java四种引用--《深入理解Java虚拟机》学习笔记及个人理解(四)

上一篇:java四种引用与回调函数


下一篇:CentOS常用命令、快照、克隆大揭秘