咱们知道java相比C,C++中没有使人头痛的指针,可是却有和指针做用类似的引用对象(Reference),就是常说的引用,好比,Object obj = new Object();这个obj就是引用,它指向的是真正的对象Object的地址,不过今天要说的是java中的四种引用。有人可能比较懵逼,四种引用?是的,从JDK1.2以后,java对引用这块的概念进行了扩充,按照引用的强度分为了四种引用:强引用,软引用,弱引用,虚引用。下面就让咱们来看看这四种引用都具体的状况吧。html
咱们平时代码中使用得最多的引用,对象的类是:StrongReference。就好比上面说的Object obj = new Object();咱们再熟悉不过了,做为最强的引用,只要引用还存在着,垃圾收集器就不会将该引用给回收,即便会出现OOM(内存溢出)。就是说这种引用只要引用还一直指向的对象,垃圾收集器是不会去管它的,因此它被称为强引用。不过若是java
Object obj = new Object(); obj = null;
obj被赋值为了null,该引用就断了,垃圾收集器会在合适的时候回收改引用的内存。
还有一种状况就是obj是成员变量,方法执行完了,obj随着被栈帧被回收了,obj引用也是一块儿被回收了。强引用的使用就不介绍了,地球人都知道。缓存
软引用是用来描述一些有用可是非必须的对象。对应的类是SoftReference,它被回收的时机是系统内存不足的时候,若是内存足够,它不会被回收,内存不足了,可能会发生OOM了,软引用的对象就会被回收。这样的特性是否是就像缓存?是的,软引用能够用来存放缓存的数据,内存足够的时候一直能够访问,内存不足的时候,须要从新建立或者访问原对象。函数
其实无论是软引用,弱引用,仍是虚引用,代码中使用方式都是像下面这样,使用对应的Reference将对象放入到构造函数当中,而后使用的地方reference.get()来调用具体对象。编码
Object obj = new Object(); SoftReference<Object> softReference = new SoftReference<>(obj); softReference.get();
同时可使用ReferenceQueue来把引用和引用队列给关联起来:.net
Object obj = new Object(); ReferenceQueue<Object> refQueue = new ReferenceQueue<>(); SoftReference<Object> softReference = new SoftReference<>(obj, refQueue);
所谓关联起来,其实就是当引用被回收的时候,会被添加到ReferenceQueue中,使用ReferenceQueue.poll()方法能够返回当前可用的引用,并从队列冲删除。简单来讲就是引用和引用队列关联起来(引用的构造函数传入队列),而后引用被回收的时候会被添加到队列中,而后使用poll()方法能够返回引用。指针
虚引用比上面两个引用就更菜了,只要垃圾收集器扫描到了它,被弱引用关联的对象就会被回收。被弱引用关联对象的生命周期其实就是从对象建立到下一次垃圾回收。对应的类是WeakReference。日志
public static void main(String[] args) throws InterruptedException { Object obj = new Object(); ReferenceQueue<Object> refQueue = new ReferenceQueue<>(); WeakReference<Object> weakRef = new WeakReference<>(obj, refQueue); System.out.println("引用:" + weakRef.get()); System.out.println("队列中的东西:" + refQueue.poll()); // 清除强引用, 触发GC obj = null; System.gc(); Thread.sleep(200); System.out.println("引用:" + weakRef.get()); System.out.println("引用加入队列了吗? " + weakRef.isEnqueued()); System.out.println("队列中的东西:" + refQueue.poll()); /** * 输出结果 * 引用:java.lang.Object@7bb11784 * 队列中的东西:null * 引用:null * 引用加入队列了吗? true * 队列中的东西:java.lang.ref.WeakReference@33a10788 */ }
能够看到当强引用被清除,手动触发GC后,弱引用回收,被加入到队列中了。code
WeakHashMap跟hashMap很像,差异就在于,当WeakHashMap的key(弱引用),指向的对象被回收了,weakhashMap中的对象也就消失了。不会和HashMap同样一直持有该对象,致使没法回收。
不赘述了,有兴趣的能够了解一下,WeakHashMap。htm
虚引用是最弱的一种引用,它不会影响对象的生命周期,对象被回收跟它没啥关系。它引用的对象能够在任什么时候候被回收,并且也没法根据虚引用来取得一个对象的实例。仅仅当它指向的对象被回收的时候,它会受到一个通知。对应的类是PhantomReference。
有人就要问既然对对象回收没影响,那它有啥用(其实用处不多),我查阅网上的资料说是,能够用来监控对象的回收,和记录日志。简单点说就是对象被回收的时候,和虚引用相关的队列知道了实例对象被回收了。这个时候咱们能够记录下来,知道对象是何时被回收的。
从而起到监控的做用。
public static void main(String[] args) throws Exception { Object abc = new Object(); ReferenceQueue<Object> refQueue = new ReferenceQueue<Object>(); PhantomReference<Object> abcRef = new PhantomReference<Object>(abc, refQueue); System.out.println("队列中的东西:" + refQueue.poll()); abc = null; System.gc(); Thread.sleep(1000); System.out.println("引用加入队列了吗? " + abcRef.isEnqueued()); System.out.println("队列中的东西:" + refQueue.poll()); /** * 输出: * 队列中的东西:null * 引用加入队列了吗? true * 队列中的东西:java.lang.ref.PhantomReference@7bb11784 */ }
发现队列中有引用了,就能够添加日志记录了。
将人比做垃圾收集器,引用比做食物,咱们来总结下四种引用:
强引用是毒药,即便你很饿了你也不会去吃它;
软引用是零食,不饿的时候不吃,饿了饥不择食,零食也能填饱肚子;
弱引用是饭菜,到了吃饭时间(垃圾回收),就吃饭菜;
虚引用是剩菜,当你吃完东西(回收完对象),就回剩下剩菜,别人就知道你吃过饭了。
引用 | 回收时机 | 使用场景 |
---|---|---|
强 | 不会被回收 | 正常编码使用 |
软 | 内存不够了,被GC | 可做为缓存 |
弱 | GC发生时 | 可做为缓存(WeakHashMap) |
虚 | 任什么时候候 | 监控对象回收,记录日志 |
1.https://blog.csdn.net/l540675...
2.https://www.iteye.com/topic/5...
3.https://www.geeksforgeeks.org...
4.https://blog.csdn.net/aitangy...