美文网首页Mvp+dagger+rxjava+retrfit+okhttp
泛型MVP解决内存泄漏隐患

泛型MVP解决内存泄漏隐患

作者: Whyn | 来源:发表于2016-12-26 21:29 被阅读2170次

    背景

    我们知道,在MVP模式中,Presenter是同时持有View和Model的引用的,那么,当在Presenter中,假设由于业务需求,需要开辟一条线程进行耗时操作,如果此时View(Activity)退出,由于子线程还在运行,此时,Presenter不能被销毁(子线程是Presenter开辟的内部类,内部类隐式持有外部类的引用),由于Presenter持有view引用,故activity即使退出了,却不能被gc,此时,内存泄露便产生了。
    eg:
    `

        public class IPresenterImpl implements IPresenter {
        private Contract.IView view;
        private Contract.IModel model;
        public IPresenterImpl(Contract.IView view) {
            this.view = view;
            model = new IModelImpl();
        }
    
        @Override
        public void loadData() {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        //模拟耗时操作
                        java.util.concurrent.TimeUnit.SECONDS.sleep(10);
                        ivew.showData();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                        ivew.onLoadDataFailed();
                    }
                }
            }).start();
        }
    }
    

    `
    ** 注: **当子线程还未运行完时,此时退出activity,则内存泄露。

    解决方法

    1.定义一个Presenter的基类:BasePresenter,BasePresenter中持有View的弱引用,并在View(Activity)的onCreate中进行attach,在View(Activity)中的onDestroy()中detach。
    `

     //V:表示任意view类型,一般是IView
    public class BasePresenter<V> {
    private WeakReference<V> weakRefView;
    public void attach(V view){
            weakRefView = new WeakReference<V>(view);
     }
    
    public void detach() {
        if(isAttach()) {
            weakRefView.clear();
            weakRefView = null;
        }
    }
    public V obtainView(){
        return isAttach()?weakRefView.get():null;
    }
    
    protected boolean isAttach() {
        return weakRefView != null &&
                weakRefView.get() != null;
     }
    }
    

    `
    ** 这里用弱引用主要时防止当Activity被销毁时没走onDestroy **(比如,kill的时候就不走)

    2.定义一个Activity基类:BaseActivity,主要就是在这里处理与BasePresenter的引用关系,后续继承这两个类的子类就不用管view与presenter绑定这些事了。

    `

    public abstract class BaseActivity<V,P extends BasePresenter<V>> 
    extends Activity{
    
    protected P mPresenter;
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mPresenter = createPresenter();
        mPresenter.attach((V)this);
    }
    
    @Override
    protected void onDestroy() {
        super.onDestroy();
        mPresenter.detach();
    }
    
    public abstract P createPresenter();
    

    }

    `
    ** createPresenter()作用是让每个Activity都产生自己需要的presenter,由自己决定。**

    综上所诉,这样便可以提高MVP的扩展性,以及解决潜在的内存泄露问题。

    具体使用方法

    通过定义一个通用的基类BasePresenter和一个BaseActivity,解决了Presenter引用View和释放时机问题,Model无需更改,IView层也无需修改,唯一需要更改点写法的就是Presenter。

    通常的Presenter接口如下所示:
    public interface IPresenter{ void startLoadData(); }
    因为所有的Presenter都要继承BasePresenter,而BasePresenter是一个抽象类,所以IPresenter也要更改为抽象类,其他的只需将其接口方法改为抽象方法即可:
    public abstract class IPresenter<V> extends BasePresenter<V>{ public abstract void startLoadData(); }
    后续的Presenter具体实现类只需继承IPresenter抽象类即可,
    public class IPresenterImpl extends IPresenter<IView>{ @Override public void startLoadData(){ //xxxxx } }

    相关文章

      网友评论

      • 情分_c2cb:为什么没看见view层啊
      • F野鸽子:Presenter不能被销毁(子线程是Presenter开辟的内部类,内部类隐式持有外部类的引用)

        假如是从线程池中开启一个子线程去做这个事情呢,那Presenter能被销毁吗?
        Whyn:我觉得,如果你是通过在Presenter内直接new Runnable(){}给到线程池,同样不能销毁呀。new Runnable(){}也是一个内部类,持有外部类Presenter的引用。只有等待线程池处理了这个Runnable,释放了对其的引用,Presenter才能被gc,从而activity才能被gc。能不能gc关键看是否有强引用指向该实例。
      • 我行不来:你好 这样写了之后 是不是在presenter中每一次调用view的方法前都得验证一下view不等于空
        if (view == null)
        return;
        Whyn:MVP中,P持有V的引用,所以在P中要调用V的方法,健壮的代码都要进行判空。
      • 唯夜:抽象类和接口怎么解决内存泄漏的问题呢,没太看明白,求大神解答。
        唯夜:@余温_8a4a 回复错啦,你该回复楼主。核心就是楼主写的这个弱引用,当activity关闭后,弱引用就释放了,同时context释放,避免内存泄漏
        d9200434285f:可以发一下源代码吗,谢谢!
        Whyn:仔细看下背景介绍,内存泄漏是因为presenter持有IView(也就是持有activity),所以定义一个基类BaseActivity,里面复写onDestroy()方法,在这里将presenter里面的IView引用置为null,那么当activity退出时,就不会因为presenter没退出而导致activity无法被gc。

      本文标题:泛型MVP解决内存泄漏隐患

      本文链接:https://www.haomeiwen.com/subject/pcdjvttx.html