Java中有四种引用类型:StrongReference(强引用),SoftReference(软引用),WeakReference(弱引用),PhantomReference(虚引用)。这四种引用的强度按照上面的顺序依次减弱。强引用就是咱们一般使用的引用类型,若是在GC Roots引用链上有一个强引用可以到达一个对象,那么这个对象就不会被垃圾回收。若是一个对象的引用为软引用,则垃圾收集器会视内存使用状况来决定是否回收对象:若是内存足够则不回收对象,否定回收。而弱引用则不相同:垃圾收集器无论内存是否足够只要一个对象只有一个弱引用,就回收对象。虚引用不会决定对象的生命周期,垃圾收集器任什么时候候均可以回收虚引用引用的对象。缓存
若是一个对象具备强引用,那垃圾回收器毫不会回收它。当内存空间不足,Java虚拟机会抛出OutOfMemoryError错误,使程序异常终止。 示例: String strongRef = new String("strong reference");
bash
若是一个对象只具备软引用,则内存空间足够,垃圾回收器就不会回收它;若是内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就能够被程序使用。软引用可用来实现内存敏感的高速缓存。spa
软引用能够和一个引用队列(ReferenceQueue)联合使用,若是软引用所引用的对象被垃圾回收器回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中。 示例:线程
// 示例1
SoftReference<String> softRef = new SoftReference<String>(new String("soft reference"));
// 示例2
ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();
SoftReference<String> softRef = new SoftReference<String>(new String("soft reference"), referenceQueue);
复制代码
弱引用与软引用的区别在于:只具备弱引用的对象拥有更短暂的生命周期。在垃圾回收器线程扫描它所管辖的内存区域的过程当中,一旦发现了只具备弱引用的对象,无论当前内存空间足够与否,都会回收它的内存。不过,因为垃圾回收器是一个优先级很低的线程,所以不必定会很快发现那些只具备弱引用的对象。code
弱引用能够和一个引用队列(ReferenceQueue)联合使用,若是弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。 实例:对象
// 示例1
WeakReference<String> weakRef = new WeakReference<String>(new String("hello ref"));
// 示例2
ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();
WeakReference<String> weakRef = new WeakReference<String>(new String("weak reference"), referenceQueue);
复制代码
“虚引用”顾名思义,就是形同虚设,与其余几种引用都不一样,虚引用并不会决定对象的生命周期。若是一个对象仅持有虚引用,那么它就和没有任何引用同样,在任什么时候候均可能被垃圾回收器回收。生命周期
虚引用主要用来跟踪对象被垃圾回收器回收的活动。虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,若是发现它还有虚引用,就会在回收对象的内存以前,把这个虚引用加入到与之 关联的引用队列中。 示例:队列
ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();
PhantomReference<String> referent = new PhantomReference<String>(new String("phantom reference"), referenceQueue);
复制代码
引用类型 | 回收时机 | 是否可能致使内存泄露 |
---|---|---|
强引用 | 不回收 | 可能 |
软引用 | 视内存状况而定 | 不可能 |
弱引用 | 回收 | 不可能 |
虚引用 | 回收 | 可能 |