MVP框架实现过程

目的:View层和Model层分离,中间由Presenter处理逻辑。

Prester作为View与Model交互的中间纽带,处理与用户交互的负责逻辑。这里我们只说明View和Presenter的交互,
Model与Presenter与前者大致相同,甚至可以直接略过接口直接使用,因此不再赘述。

我们期望减少在Activity,Fragment类的容量,使View层专心View的实现,将逻辑由P层处理。

即存在对View实现的V和对逻辑实现的P,V中调用P的方法处理逻辑,P调用V处理View变化。那么怎样将两者联系在一起?

实现:

1、建立联系

如果想要在V中调用P的方法,只要在V中建立P的对象。即在V中,P p = new P(); 然后p.xx();

此时只是V单向的含有P的引用,让p拥有v的引用。理所当然,我们通过p.attachView(this); 传递this参数,
使p对象拥有对应的v对象的引用。

同时,不能忘记使用p.detachView(); 目的是保持P和V的生命周期一致,即在V销毁时,P逻辑上也失去效用,也应销毁,
防止由于P保持对V的引用,导致V无法销毁,内存溢出。

注:对于内存溢出,我们可以在P中attchView()和detachView()具体实现中使用弱引用的方式,强化防止内存溢出。


public class MvpView {

    private MvpPresenter presenter;

    public void onCreate(){

        presenter = new MvpPresenter();

        presenter.attachView(this);
    }

    public void onDestroy(){

        presenter.detachView();
    }
}

public class MvpPresenter {

    private WeakReference<MvpView> viewRef;

    public void attachView(MvpView view) {
        viewRef = new WeakReference<>(view);
    }

    public void detachView() {
        if (viewRef != null) {
            viewRef.clear();
            viewRef = null;
        }
    }

    public void xx() {}
}

但是这里远远没有结束,我们不能忽略xx()。也就是


public void attachView(MvpView view) {
    viewRef = new WeakReference<>(view);
}

MvpView不能写死,因此需要使用接口,只是声明方法,无需关心方法的具体实现。(本人始终觉得接口最好用的在于分层后,
Coder只需先关注于一层的逻辑,而无需在几层之间来回切换)由于下面将接口和连接的实现分离,这里不重复给出源码。

2、实现复用(继承,泛型)

由于并不是只有某个View代码需要分离,同时实现连接的共同方法可以复用,这里我们需要实现V和P建立联系的实现的复用。

步骤有二,一是将不同View接口和实现连接的公共部分分离,公共部分抽离成父类,二是使得match的特定V和P相互联系。

1)如果单纯的抽离成父类,这十分简单,只需要将上述的代码直接copy到工程的BaseActivity或者BaseFragment中,
或者单独隔离成继承Base类的MvpBase类。

无疑我们采取单独隔离成继承Base类的MvpBase类的方式,既是因为实现逻辑的分离,同时并不是所有View逻辑都适合分离出,
也是因为为了下一步使得match的特定V和P相互联系。

2)这时假设我们有Va和Vb,都继承MvpView;同时有Pa和Pb,也都继承MvpPresenter。很显然这样V父类中的实现无法知晓该返回
Pa或者Pb的对象,同样的MvpPresenter也只能对父类MvpView对象操作。

自然而然我们想到了泛型,在父类中添加泛型,在类被使用时再被告知该返回哪个子类。由下文代码找到P中需要为泛型是接口IViewA。
而V需要泛型的为Presenter类。

public class MvpViewA implements IViewA {

    private MvpPresenter presenter;

    public void onCreate(){

        presenter = new MvpPresenter();

        presenter.attachView(this);
    }

    public void onDestroy(){

        presenter.detachView();
    }

    public void doSomeInV(){}
}

public class MvpPresenter {

    private WeakReference<IViewA> viewRef;

    public void attachView(IViewA view) {
        viewRef = new WeakReference<>(view);
    }

    public void detachView() {
        if (viewRef != null) {
            viewRef.clear();
            viewRef = null;
        }
    }

    public void doSomeInP(){
        viewRef.doSomeInView()
    }
}

变为


public abstract class MvpBasePresenter<V extends IMvpView> implements MvpPresenter<V> {

    private WeakReference<V> viewRef;

    @Override
    public void attachView(V view) {
        viewRef = new WeakReference<V>(view);
    }

    protected V getView() {
        return viewRef.get();
    }
    @Override
    public void detachView() {
        if (viewRef != null) {
            viewRef.clear();
            viewRef = null;
        }
    }
}


public abstract class MvpBaseView<P extends MvpPresenter> extends BaseView implements IMvpView {
    protected P presenter;

    protected abstract P createPresenter();

    protected void onCreate() {
        presenter = createPresenter();

        presenter.attachView(this);
    }

    protected void onDestroy() {
        presenter.detachView();
    }
}

当然MVP的实现会根据每个Coder的需求,进行不同的设计,菜鸟一枚的我就不拓展了。