1、强引用(StrongReference)
强引用就是平时最常用的定义变量的方式,即:
val strongReference="强引用"
val strongReferenceList= mutableListOf<String>("强","引","用")
- 强引用可以直接访问目标对象。
- 强引用所指向的对象在任何时候都不会被系统回收。当内存空间不足时,Java虚拟机宁可抛出OutOfMemoryError异常,也不会通过去回收强引用所指向的对象来释放内存。
- 强引用可能导致内存泄露。
2、软引用(SoftReference)
软引用就是在原先定义一个对象的条件下,再将这个对象装进SoftReference中,通过调用SoftReference中的get()函数,可以拿到我们装入SoftReference的对象。
val softReference=SoftReference("软引用")
softReference.get()
如果软引用指向的对象被回收了,那么get()函数将会返回null值,所以在使用时,需要判断一下获取的对象是否为null值,避免出现NullPointerException异常导致程序崩溃。
-
- 如果一个对象只具有软引用,那么如果内存足够,软引用指向的对象不会被回收,调用get()函数会返回当前指向的对象;当内存不足时,则会被回收,调用get()函数则返回null值。
- 软引用可用来实现内存敏感的高速缓存。
- 软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所指向的对象被回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列ReferenceQueue中。
引用队列(ReferenceQueue)
val referenceQueue: ReferenceQueue<String> = ReferenceQueue()
val softReference=SoftReference("软引用",referenceQueue)
- 当软引用指向的对象被回收后,此软引用将会被传入referenceQueue引用队列中(referenceQueue引用队列中存储的都是被回收了的引用)。
- 当我们调用引用队列的poll()方法的时候,如果这个队列中不是空队列,那么将返回队列前面的那个Reference对象;如果队列为空,那么将返回一个null值。
引用队列的作用:用于检查引用所指向的对象是否已经被回收了。并进行一些简单操作。
3、弱引用(WeakReference)
- 如果一个对象只具有弱引用,那么当垃圾回收器线程扫描的过程中,一旦发现了只具有弱引用的对象,不论当前内存是否足够,它都会立即把这个对象给回收掉。
- 但是由于垃圾回收器线程优先级是很低的,因此不会立即扫描到这个只具有弱引用的对象,会有延迟。
val weakReference=WeakReference("弱引用")
weakReference.get()
4、虚引用(PhantomReference)
- 形同虚设,虚引用主要用来跟踪对象被垃圾回收器回收的活动。
- 虚引用和其他引用不同的地方是:虚引用必须跟引用队列ReferenceQueue一起使用
- 当垃圾回收器发现一个只具有虚引用的对象时,就会把这个对象回收并将其虚引用加入到引用队列中
val phantomQueue = ReferenceQueue<String>()
val phantomReference=PhantomReference("虚引用", phantomQueue)
总结
- 优先级:强引用(不会被回收)> 软引用(内存不足时回收) > 弱引用(随时回收) > 虚引用(随时回收,基本不使用)
- 当程序出现内存泄漏或内存溢出时,则可以考虑使用软引用或虚引用。
- 虚引用只是为了排查对象被回收的活动。可以理解为debug?
- 当一个对象既有强引用,又有其他引用时,则以优先级最高的强引用为准,不会被回收。其他情况也以此类推。
在安卓中,Activity A给另一个组件B传入当前Activity A的Context,那么组件B则得到了Activity A的引用;当Activity A被销毁、且组件B不需要被销毁时,由于组件B对Activity A的context存在强引用,那么这个Activity A将得不到及时的销毁,这样就造成了内存泄漏。
|