前言
说来惭愧,最近在面试中在JVM基础上面又栽跟头了,尤其是在引用这方面,之前觉得引用这里比较晦涩难懂,枯燥乏味,每次看到JVM中的引用部分内容时,总是会选择性地跳过,所以看起来这次的跌倒是注定的,好,那么废话不多说,我们今天就来尝试弄懂引用这一块内容
何谓引用?
如果reference类型的数据中存储的数值是另一块内存的起始地址,那么这块内存就代表着一个引用。
这种定义就比较纯粹,因为他只能代表被引用或者不被引用两种状态,对于一些“食之无味,弃之可惜”的对象就无能为力了,也就是说我们希望能够描述这样的一类对象:内存空间还足够的时候,则这些对象还可以保存在内存之中,如果内存空间在进行垃圾回收之后还是很紧张,则可以抛弃这些对象,很多系统的缓存功能都符合这种场景,所以在jdk1.2之后,java对引用的概念进行了扩充,将引用分为强引用、弱引用、软引用、虚引用4种,4中引用强度依次减弱
(出自深入理解Java虚拟机)
强引用
强引用就是指程序代码中普遍存在的,类似“Object o = new Object();”这类的引用,只要强引用还存在,垃圾回收器永远不会回收掉被引用的对象,即便内存不足的时候,虚拟机宁愿抛出内存不足的异常,也不会去回收这些对象
使用场景
大部分使用场景都是使用了强引用,比如使用new关键字创建对象,通过反射的方式获得一个对象
软引用
软引用是用来描述一些还有用,但是非必需的对象。对于软引用所关联着的对象,在系统将要发生内存溢出异常之前,将会把这些对象列进回收范围之中进行第二次回收,如果这次回收还是没有足够的内存,才会抛出内存溢出异常,jdk1.2之后,提供了SoftReference类来实现软引用
使用场景
有可能在创建后使用的对象,也可能不使用,所以这里基于内存的考虑会使用软引用,常见于缓存方面的使用
弱引用
弱引用就是用来描述非必需对象的,但是他的强度比软引用更弱一些,被弱引用关联的对象只能生存到下一次垃圾收集发生之前。当垃圾收集器工作的时候,无论当前内存是否足够,都会回收掉只被弱引用所关联的对象,jdk1.2之后,提供了WeakRefrence类来实现弱引用
使用场景
弱引用作用于生命周期更短的,对内存更为敏感的场景当中,只要一发生内存回收就会回收掉这部分内存,比如占用内存较大的Map,WeakHashMap,ThreadLocal中的ThreadLocalMap中的key都是弱引用
1 | String str = "hello weak reference"; |
虚引用
虚引用也被称为幽灵引用或者幻影引用,它是最弱的一种引用关系,一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚引用来取得一个对象实例。为一个对象设置虚引用关联的唯一目的就是能在这个对象被收集器回收时收到一个系统通知。也就是在虚引用锁描述的对象被回收之前,会被JVM放入ReferenceQueue中(其他引用是在引用的对象被销毁之后才被传入ReferenceQueue中),由于这个机制的存在,虚引用大多数被用于引用销毁前的处理工作,另外,虚引用在创建的时候必须带有ReferenceQueue
在jdk1.2之后,提供了PhantomReference类来实现虚引用
1 | PhantomReference<String> pref = new PhantomReference<String>(new String("str"),new ReferenceQueue()); |
使用场景
对象销毁前的一些操作,比如资源释放等