JVM学习–强引用,软引用,弱引用,虚引用
强引用
强引用就是代码中通过常规方法创建的对象,比如new,反序列化,反射等方式创建的对象。只要强引用还存在,那么垃圾收集器就不会回收该对象。
软引用
如果一个对象只具有软引用,则内存空间足够,垃圾回收器就不会回收它;如果内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用。软引用可用来实现内存敏感的高速缓存(下文给出示例)。
软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被垃圾回收器回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中。
Java中使用SoftReference类来实现。
弱引用
弱引用与软引用的区别在于:只具有弱引用的对象拥有更短暂的生命周期。在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。
弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。
Java中使用WeakReference类来实现。
虚引用
“虚引用”顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。
虚引用主要用来跟踪对象被垃圾回收器回收的活动。虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之 关联的引用队列中。
Java中使用PhantomReference类来实现。
例子
public class ReferenceDemo {
private String name;
public ReferenceDemo(String name) {
this.name = name;
}
public static void main(String[] args) {
//创建强引用对象
ReferenceDemo demo1 = new ReferenceDemo("AA");
ReferenceDemo demo2 = new ReferenceDemo("BB");
ReferenceDemo demo3 = new ReferenceDemo("CC");
ReferenceDemo demo4 = new ReferenceDemo("DD");
ReferenceDemo demo5 = new ReferenceDemo("FF");
ReferenceDemo demo6 = new ReferenceDemo("EE");
//关联软引用
SoftReference<ReferenceDemo> soft1 = new SoftReference<ReferenceDemo>(demo1);
SoftReference<ReferenceDemo> soft2 = new SoftReference<ReferenceDemo>(demo2);
//关联弱引用
WeakReference<ReferenceDemo> soft3 = new WeakReference<ReferenceDemo>(demo3);
WeakReference<ReferenceDemo> soft4 = new WeakReference<ReferenceDemo>(demo4);
//关联虚引用
ReferenceQueue<ReferenceDemo> queue = new ReferenceQueue<ReferenceDemo>();
PhantomReference<ReferenceDemo> soft5 = new PhantomReference<ReferenceDemo>(demo5,queue);
// PhantomReference<ReferenceDemo> soft6 = new PhantomReference<ReferenceDemo>(demo6);
//去除强引用
demo1 = null;
demo2 = null;
demo3 = null;
demo4 = null;
System.gc();
print(soft1);
print(soft2);
print(soft3);
print(soft4);
print(soft5);
}
public static void print(Reference<ReferenceDemo> reference){
ReferenceDemo demo = (ReferenceDemo)reference.get();
if(demo == null){
System.out.println(reference.getClass()+" value = null");
}
else{
System.out.println(reference.getClass()+" value = " + demo.getName());
}
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
当 demo3 = null; demo4 = null; 不存在时
输出
class java.lang.ref.SoftReference value = AA
class java.lang.ref.SoftReference value = BB
class java.lang.ref.WeakReferenceis value = CC
class java.lang.ref.WeakReferenceis value = DD
class java.lang.ref.PhantomReferenceis value = null
当 demo3 = null; demo4 = null; 存在时
输出
class java.lang.ref.SoftReference value = AA
class java.lang.ref.SoftReference value = BB
class java.lang.ref.WeakReferenceis value = null
class java.lang.ref.WeakReferenceis value = null
class java.lang.ref.PhantomReferenceis value = null
从上面可以看出
- 虚引用关联的对象是无法获取到的。
- 软引用关联的对象在垃圾回收时,内存空间足够,垃圾回收器就不会回收它。
- 弱引用关联的对象在垃圾回收时,如果对象的强引用不存在,不管内存空间够不够,都会被回收。