Android开发了解:强引用、软引用、弱引用、虚引用

2022-04-01  本文已影响0人  Junker_
  1. 强引用是开发过程中最常用的引用方式,当一个对象具有强引用时,操作系统进行 GC 回收处理是不会回收强引用的对象,即使系统内存不足,Java虚拟机宁可抛OutOfMemoryError(内存溢出错误),宁可使程序异常终止,也不会靠回收强引用的对象来解决内存不足的问题。
  2. 只要把强引用对象 str 赋空值 null, 该对象就可以被 GC 垃圾回收器回收;因为该对象此时不再含有其他强引用。
用法示例:
  // 变量 str 表示强引用,指向 new String("junker") 这个对象
  String str = new String("junker"); 
  1. 当JVM虚拟机内存充足时,软引用对象不会被 GC 垃圾回收器回收。
  2. 当JVM虚拟机内存不足时,软引用对象会被 GC 垃圾回收器回收。
  3. 未被回收的软引用对象是一直会被程序占有的。
用法示例:
  MySoftObj softObj = new MySoftObj();
  //软引用实例
  SoftReference softRef = new SoftReference(softObj);
  //获取软引用保存的引用
  MySoftObj anotherRef = (MySoftObj) softRef.get();
  1. 软引用可以和引用队列(ReferenceQueue)联合使用来实现内存紧张的高速缓存;如果软引用引用的对象被回收,Java虚拟机会把改软引用对象加到与之关联的引用队列中。
用法示例:
  MySoftObj softObj = new MySoftObj();
  ReferenceQueue queue = new  ReferenceQueue();
  SoftReference  softRef = new  SoftReference(softObj, queue);
用法示例:
  MyWeakObj weakObj = new MyWeakObj();
  //弱引用实例 
  WeakReference weakReference = new WeakReference<>(weakObj); 
  //获取弱引用保存的引用 
  MyWeakObj anotherRef = weakReference.get(); 
  1. 对于弱引用对象,当操作系统进行 GC 回收处理时,不管内存空间是否足够,弱引用对象都会被回收。
  2. 如果一个对象除了具有弱引用还具有强引用,GC回收时,该对象是不会被回收的,操作系统只会回收只具有弱引用的对象。
  3. 弱引用常常被用于防止内存泄漏,最常见的是单例和Handler造成的内存泄漏。

相关代码实现:

 /**
     * 将匿名内部类对象,定义成全局变量,
     * 这样 baseCallBack 的生命周期就和 外部Activity 一样
     */
    protected BaseCallBack baseCallBack;

    private void startLongTimeRequest() {
        baseCallBack = new BaseCallBack() {
            @Override
            public void onSuccess(String data) {
                //执行业务逻辑
            }
        };
        //执行异步耗时请求
        CustomManager.getInstance().requestApi(new BaseCallBackWeak(baseCallBack));
    }

    static class BaseCallBackWeak implements BaseCallBack {
        private WeakReference<BaseCallBack> backWeakReference;

        public BaseCallBackWeak(BaseCallBack callback) {
            this.backWeakReference = new WeakReference<>(callback);
        }

        @Override
        public void onSuccess(String data) {
            //判断弱引用对象是否被回收
            if (backWeakReference != null && backWeakReference.get() != null) {
                backWeakReference.get().onSuccess(data);
            }
        }
    }

1、虚引用不能保证其保存对象生命周期,若保存对象只有虚引用,则其有效期完全随机于GC的回收,在任何一个不确定的时间内,都可能会被回收;而虚引用与其他几者的引用不同在于,在使用PhantomReference,必须要和Reference联合使用。

用法示例:
  MyPhantomObj phantomObj = new MyPhantomObj();
  //引用队列 
  ReferenceQueue queue = new ReferenceQueue<>(); 
  //虚引用 
  PhantomReference phantomReference = new PhantomReference(phantomObj, queue); 
  //获取虚引用保存的引用 
  MyPhantomObj anotherRef = phantomReference .get(); 
上一篇 下一篇

猜你喜欢

热点阅读