优雅的处理LiveData数据倒灌并解决内存泄露的问题

文章讨论了原始MutableLiveData在处理数据变化时可能导致内存泄露的问题,提出通过TimeSequenceLiveData类来替代,以确保Observer的正确移除,从而避免内存泄漏。
摘要由CSDN通过智能技术生成

网上给的防止数据倒灌的很多都有内存泄露风险,原因在于Observer对象变了,在removeObservers的时候就无法把Observer移除掉,在多次调用observe之后就会出现内存泄露问题
下面的方案修改面很小,只需要用TimeSequenceLiveData替换掉MutableLiveData就好了,源码如下:

public class TimeSequenceLiveData<T> extends MutableLiveData<T> {
    private long mLastSetValueTime;
    //key为newObserver,value为包装的Observer
    private final ConcurrentHashMap<Observer<? super T>, ObserverWrapper<T>> mObserverMap = new ConcurrentHashMap<>();

    public TimeSequenceLiveData() {
    }

    public TimeSequenceLiveData(T value) {
        super(value);
    }

    @Override
    public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
        Observer<T> newObserver = new Observer<T>() {
            private final long initTime = System.currentTimeMillis();

            @Override
            public void onChanged(@Nullable T t) {
                if (initTime < mLastSetValueTime) {
                    observer.onChanged(t);
                }
            }
        };
        mObserverMap.put(newObserver, new ObserverWrapper<>(owner, newObserver));
        super.observe(owner, newObserver);
    }

    @Override
    public void removeObservers(@NonNull LifecycleOwner owner) {
        super.removeObservers(owner);
        HashSet<Observer<? super T>> removeKeySet = new HashSet<>();
        for (Observer<? super T> observer : mObserverMap.keySet()) {
            ObserverWrapper<T> tObserverWrapper = mObserverMap.get(observer);
            if (null == tObserverWrapper) {
                continue;
            }
            if (tObserverWrapper.mLifecycleOwnerWeakReference.get() == owner) {
                super.removeObserver(tObserverWrapper.mWrapperObserver);
                removeKeySet.add(observer);
            }
        }
        for (Observer<? super T> observer : removeKeySet) {
            mObserverMap.remove(observer);
        }
    }

    @Override
    public void setValue(T value) {
        mLastSetValueTime = System.currentTimeMillis();
        super.setValue(value);
    }

    @Override
    public void removeObserver(@NonNull Observer<? super T> observer) {
        ObserverWrapper<T> tObserverWrapper = mObserverMap.get(observer);
        if (null != tObserverWrapper) {
            super.removeObserver(tObserverWrapper.mWrapperObserver);
            mObserverMap.remove(observer);
        } else {
            super.removeObserver(observer);
        }
    }

    private static class ObserverWrapper<T> {
        @NonNull
        private final WeakReference<LifecycleOwner> mLifecycleOwnerWeakReference;
        @NonNull
        private final Observer<? super T> mWrapperObserver;

        public ObserverWrapper(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer) {
            mLifecycleOwnerWeakReference = new WeakReference<>(owner);
            mWrapperObserver = observer;
        }
    }
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值