目的 :
- 可以让程序员通过代码的方式来决定某个对象的生命周期
- 有利于垃圾回收
1. 强引用
强引用是使用最普遍的引用, 我们写的代码 99.9999% 都是强引用, 只要某个对象有强引用与之关联, 这个对象永远不会被回收, 即使内存不足, JVM 宁愿抛出 OOM 也不会去回收。
在实际开发中,若看到有对象被手动赋值为null,很大可能就是为了“特意提醒”JVM这块资源可以进行垃圾回收了。
2. 软引用
只有在内存不足时,JVM 才会回收该对象。当内存不足时,会触发 JVM 的GC,如果 GC 后内存还是不足,就会把软引用包裹的对象给干掉。
软引用主要用于缓存:当内存足够时,可以正常的拿到缓存,当内存不够时,就会先干掉缓存,不至于马上抛出OOM。 比如:浏览器的后退按钮。按后退时,这个后退时显示的网页内容是重新进行请求还是从缓存中取出呢?这就要看具体的实现策略了。
如果一个网页在浏览结束时就进行内容的回收,则按后退查看前面浏览过的页面时,需要重新构建;
如果将浏览过的网页存储到内存中会造成内存的大量浪费,甚至会造成内存溢出。
3. 弱引用
不管内存是否足够,只要发生GC,弱引用就会被回收。
ThreadLocal 的 key 使用弱引用从而避免内存泄漏
4. 虚引用
无法通过虚引用来获取对一个对象的真实引用。虚引用必须与ReferenceQueue一起使用。当GC准备回收一个对象时,如果发现它还有虚引用,就会在回收之前,把这个虚引用加入到与之关联的ReferenceQueue中。
虚引用主要用来跟踪对象被垃圾回收器回收的活动。
程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要进行垃圾回收。如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取必要的行动。
NIO运用了虚引用管理堆外内存
.