MVP基类抽取,面向接口

通常项目中使用MVP架构并不会像MVP基础结构Demo中这样直接把类写死,而是要进行基类抽取,面向接口。

创建三个基类:BaseModelBaseViewBasePresenter

MVP整体流程是这样的:

  1. 用户操作View
  2. View把任务传递给Presenter
  3. Presenter把任务传递给Model
  4. Model执行具体的任务,任务完成后把结果返回给Presenter
  5. Presenter把结果返回给View

根据这个流程,按View -> Presenter -> Model 这个顺序去梳理基类,思路更清晰。

首先是BaseView,这里有两点需要完成:

  1. 根据上述流程第二步描述,View里肯定是需要持有Presenter,这个Presenter具体是哪个类只有到具体业务才知道,所以这个Presenter的赋值需要由具体的子类实现。
  2. 根据上述流程第五步描述,Presenter需要持有View,并且Presenter和View之间需要有个协议,Presenter才能把结果返回给View,也就是说View必须提供出一个方法给Presenter调用。这里的具体协议也需要具体到业务才能确定,所以也需要由子类实现.

综上两点,BaseView代码如下:

//这里继承Activity,当然也可以是其他Fragment或者其他View
public abstract class BaseView<P extends BasePresenter, CONTRACT> extends Activity {

    protected P p;

    @Override
    protected void onCreate( Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        p = getPresenter();
        p.v = this;//这里会产生内存泄漏,不影响现在理解,稍后解决
    }

    // 这个Presenter具体是哪个类只有到具体业务才知道,所以这个Presenter的赋值需要由具体的子类实现
    public abstract P getPresenter();

    // 获取具体协议
    // 协议规定了Presenter怎么和View沟通
    public abstract CONTRACT getContract();
}

然后需要完成BasePresenter,上述MVP流程,Presenter需要分别持有View和Model,Presenter是View和Model沟通的桥梁,把操作从View从递给Model,把结果从Model返回给View,而Presenter与他们之间的沟通,也是要遵循某种协议。
Presenter的View在BaseView里通过p.v = this已经赋值,这里指需要给Model赋值,而具体的Model也需要由具体业务的子类去完成,不在累赘:

public abstract class BasePresenter<M extends BaseModel, V extends BaseView, CONTRACT> {

    protected M m;
    protected V v;

    public BasePresenter() {
        m = getModel();
    }

    public abstract M getModel();
    
    public abstract CONTRACT getContract();
}

最后是完成BaseModel,Model里需要持有Presenter,并且和Presenter之间也有协议,具体的赋值和协议和上述思想一样,都交给具体子类:


public abstract class BaseModel<P extends BasePresenter, CONTRACT> {

    protected  P p;

    // 子类的初始化方法调用super.BaseModel(xxxPresenter),就会实现具体的Presenter和Model的绑定
    public BaseModel(P p) {
        this.p = p;
    }

    public abstract CONTRACT getContract();

}

三个基类完成,模拟一个登录业务爽一下:
一般情况下网络请求都会有个BaseBean,这里定义为BaseResponseBean

public class BaseResponseBean {

    private int code;

    private String error;

    public int getCode() {
        return code;
    }

    public void setCode(int code) {
        this.code = code;
    }

    public String getError() {
        return error;
    }

    public void setError(String error) {
        this.error = error;
    }
}

然后定义一个UserInfo用来接收网络请求返回的数据,继承自BaseRequrieResult


public class UserInfo extends BaseResponseBean {

    private String nickName;

    public UserInfo(String nickName) {
        this.nickName = nickName;
    }

    public String getNickName() {
        return nickName;
    }

    public void setNickName(String nickName) {
        this.nickName = nickName;
    }
}

之后就要定义一直提到的协议,上面流程也分析过,Model和Presenter需要有执行操作的函数,Presenter和View需要有接收结果的函数:

// 数字流程顺序,方便理解
public interface LoginContract {

    interface View<T extends BaseResponseBean> {
        void responseLogin(T t);//-------------------------------------4
    }

    interface Presenter<T extends BaseResponseBean> {
        void requireLogin(String userName, String password);//---------1
        void responseLogin(T t);//-------------------------------------3
    }

    interface Model {
        void requireLogin(String userName, String password);//---------2
    }

}

协议完成,现在可以去写具体业务了:LoginModelLoginActivityLoginPresenter分别继承自:BaseModelBaseViewBasePresenter
View层这里就是Activity,因为BaseView技能了Activity,这里LoginActivity就直接继承BaseView,布局很简单不贴了,直接上LoginActivity代码:

public class LoginActivity extends BaseView<LoginPresenter, LoginContract.View> {

    private EditText et_username;
    private EditText et_pwd;
    private Button button;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_login);

        initView();
        initOnclickListener();
    }

    private void initView() {
        et_username = findViewById(R.id.et_username);
        et_pwd = findViewById(R.id.et_pwd);
        button = findViewById(R.id.btn_login);
    }

    private void initOnclickListener() {
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                // p已通过getPresenter赋值
                p.getContract().requireLogin(et_username.getText().toString(), et_pwd.getText().toString());
            }
        });
    }

    @Override
    public LoginPresenter getPresenter() {
        return new LoginPresenter();
    }

    @Override
    public LoginContract.View getContract() {
        return new LoginContract.View<UserInfo>() {
            @Override
            public void responseLogin(UserInfo userInfo) {
                Toast.makeText(LoginActivity.this, userInfo == null ? "登录失败" : "登录成功:" + userInfo.getNickName(), Toast.LENGTH_LONG).show();
            }
        };
    }

}

然后是Presenter,这里有三种写法:

  1. 在Presenter里处理的具体的操作(Google的Demo里是这样做的)
  2. 用其他的业务类去做具体操作
  3. 在Model层去做具体操作
    这里采用的第三种,把任务给了Model层。这个看团队选择和个人习惯,没什么好坏。
    LoginPresenter里上面都没做,只是分发:
public class LoginPresenter extends BasePresenter<LoginModel, LoginActivity, LoginContract.Presenter> {


    @Override
    public LoginModel getModel() {
        return new LoginModel(this);
    }

    @Override
    public LoginContract.Presenter getContract() {
        return new LoginContract.Presenter<UserInfo>() {
            @Override
            public void requireLogin(String userName, String password) {
                // 任务发送给Model
                m.getContract().requireLogin(userName, password);
            }

            @Override
            public void responseLogin(UserInfo userInfo) {
                // 结果返回给Presenter
                v.getContract().responseLogin(userInfo);
            }
        };
    }
}

然后是LoginModel,在这做具体的登录操作,简单模拟了一下,并没有做真正的网络请求:

public class LoginModel extends BaseModel<LoginPresenter, LoginContract.Model> {


    public LoginModel(LoginPresenter loginPresenter) {
        // 此时通过父类的构造方法把具体的loginPresenter赋值给了BasePresenter中的p
        super(loginPresenter);
    }

    @Override
    public LoginContract.Model getContract() {
        return new LoginContract.Model() {
            @Override
            public void requireLogin(String userName, String password) {
                // 结果返回给Presenter
                if ("123".equals(userName) && "123".equals(password)){
                    p.getContract().responseLogin(new UserInfo("yu"));
                }else {
                    p.getContract().responseLogin(null);
                }
            }
        };
    }
}

以上就完成了View->Presenter->Model->Presenter->View这么一个完整的流程。
上面大码用了很多泛型,一定要写好泛型之后再用提示自动生成代码,不亲手撸一把体会不到有多爽,完全不需要强转什么的,谁爽谁知道!!!

代码到这里并没有完事,真是项目中的登录操作要做网络请求,这里如果处理耗时操作,就会出现内存泄漏,在LoginModelrequireLogin方法里模拟一个耗时操作:

            @Override
            public void requireLogin(String userName, String password) {
                new Thread(){
                    @Override
                    public void run() {
                        super.run();
                        SystemClock.sleep(50000);
                    }
                }.start();
            }

run一把,点击登录,然后退出Activity,打开Profiler看MEMORY:

LoginActivity泄漏了我的哥

是的,LoginActivity泄漏了,原因就是Model 中持有Presenter的引用,Presenter中持有LoginActivity的引用,而Model中有耗时操作没有销毁。
解决方式就是把Presenter对View的引用变成弱引用,修改BasePresenter,如下:

public abstract class BasePresenter<M extends BaseModel, V extends BaseView, CONTRACT> {

    ......
//    不能才用强引用,会发送内存泄漏
//    protected V v;

//    弱引用
    private WeakReference<V> vWeakReference;

//    弱引用绑定
    public void bindView(V v) {
        vWeakReference = new WeakReference<V>(v);
    }

//    弱引用解绑,回收
    public void unBindView() {
        if (vWeakReference != null) {
            vWeakReference.clear();
            vWeakReference = null;
            System.gc();
        }
    }

//    提供弱引用获取方法
    public V getView(){
        if (vWeakReference != null) {
            return vWeakReference.get();
        }
        return null;
    }

    ......
}

BaseView里也不能直接用p.v = this;这样赋值了,而是采用绑定的方式,并且在onDestroy里解绑:

public abstract class BaseView<P extends BasePresenter, CONTRACT> extends Activity {

    protected P p;

    @Override
    protected void onCreate( Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        p = getPresenter();
        // 强引用会导致内存泄漏
        //  p.v = this;
        p.bindView(this);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 和Presenter解绑
        p.unBindView();
    }

    ......
}

Presenter里不再持有View强引用,LoginPresenter里也不能通过v.getContract().responseLogin(userInfo);去调用View层,而是要通过getView()去获取v:

public class LoginPresenter extends BasePresenter<LoginModel, LoginActivity, LoginContract.Presenter> {


    @Override
    public LoginModel getModel() {
        return new LoginModel(this);
    }

    @Override
    public LoginContract.Presenter getContract() {
        return new LoginContract.Presenter<UserInfo>() {
            ......
            public void responseLogin(UserInfo userInfo) {
                // 结果返回给Presenter
                // Presenter不在持有v,不能通过v.getContract().responseLogin(userInfo)去调用,通过getView()获取对View的弱引用;
                getView().getContract().responseLogin(userInfo);
            }
        };
    }
}

哦了,再看看memory:


LoginActivity不见了哦。
如果是MVC,Activity就是Controller,耗时操作都直接在Activity里执行,就会发生内存泄漏。

MVP基础结构Demo
项目地址

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容