美文网首页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