1. Java中的四种引用类型
在Java中,对于引用最基本的解释就是:如果reference类型的数据中存储的数值代表的是另外一块内存的起始地址,就称这块内存代表着一个引用(有点指针的意味)。后来Java还将引用划分为了4种,根据被GC回收的时机可以分为:强引用(Strong Reference)、软引用(Soft Reference)、弱引用(Weak Reference)、虚引用(Phantorm Reference)。这4种引用的强度依次渐弱。
1.1 强引用
Object o = new Object();
这里的o
就是一个强引用,也是我们用得最多的引用,在实例化类的时候经常会用到。遇到这类引用,GC(垃圾回收器)是绝对不会回收它的。当遇到内存不足的情况,JVM会抛出OOM异常。所以,在不使用这类对象的时候要注意释放它,以便让系统回收。
1.2 软引用
SoftReference<String> s = new SoftReference<>(new String("Hello"));
System.out.println(w.get());
此时会输出
Hello
这里的s
就是一个软引用,它是用来描述一些有用但非必需的对象,当系统内存出现不足的时候,会立即把这些对象进行回收。
软引用可以用来实现内存的缓存,如图片缓存。
1.3 弱引用
WeakReference<String> w = new WeakReference<>(new String("Hello"));
System.out.println(w.get());
System.gc();
System.out.println(w.get());
此时会输出:
Hello
null
弱引用的强度比软引用低一些,一旦GC开始执行,便会对这类对象进行回收。也就是说无论内存足够与否,弱引用对象都只能生存到下一次GC之前。
1.4 虚引用
ReferenceQueue<String> queue = new ReferenceQueue<>();
PhantomReference<String> pr = new PhantomReference<>(new String("Hello"), queue);
System.out.println(pr.get());
此时会输出:
null
虚引用也称为幽灵引用,它的强度是最低的,一个对象是否有虚引用的存在,完全不会对它的生存时间构成影响,当然也无法通过虚引用获取一个类的实例,GC任何时候都有可能会回收此对象。它唯一的作用就是就是用于追踪,让我们能够在这个对象被回收的时候收到一个通知。
2.如何判断对象需要被回收
2.1 引用计数法
比较尴尬的是以前刚上Java课的时候,我去问老师JVM是如何确定需要被回收的对象的时候,老师就给我讲了引用计数法,但后来随着学习的深入,我发现主流的JVM都没有采用这一方法,原因是它有很大的一个缺陷,就是难以解决对象之间相互引用问题。
所谓引用计数法,就是在对象内部会有一个引用计数器,一旦某个地方引用它时,计数器就加1,表面上看这是一个效率非常高的方式,但是如果遇到如下情形时,采用了这种方式判断对象是否存活的GC就难以发挥作用了。
public static void main(String[] args) {
A a = new A();
B b = new B();
//a,b循环引用对方
a.intance = b; //b的引用计数加1
b.intance = a; //a的引用计数加1
//虽然a和b都被设置成null了,但它们的引用计数仍然是1,不会被GC回收