美文网首页
小白用Android MVP-初体验(一)

小白用Android MVP-初体验(一)

作者: DuBetter | 来源:发表于2019-03-29 16:34 被阅读0次

    写android以来,一直都是采用MVC的模式,所有的业务逻辑,网络请求等都放在了View中,即Activity或者Fragment中。看了一些mvp文章,很多跨度较大,自己造诣不够,不能跟上大神们的思路。今天自己初次尝试MVP,准备一步一步慢慢来,这是第一篇-初体验,用我们常用的熟悉的登录业务引入;

    介绍

    MVC

    M:逻辑模型,V:试图模型,C:控制器, 耦合性强,不利于后期维护,适合在一些较小的项目中使用;

    MVP

    Model: 具体的数据源和数据请求,访问数据库和后台服务器等;
    View:负责UI,Activity,Fragment;
    Presenter:是View和Model的桥梁,负责接收View发来的指令,并据此调用model中的方法;

    • 释放了View,作为View 来说只需要负责UI。耦合度降低。
    • 引入了Presenter,会导致其越来越臃肿,同样不利于后期的维护,并且,每一个包含网络请求的View都需要对应一个或者多个Presenter。

    实现

    登录活动(Activity)

    很简单,点击登录执行登录过程;


    界面
    DataModel

    首先来看Model层,一般Model层需要和数据库或者后台服务器交互,这里模拟即可。

    package com.mypractisedemos.module.mvpdemo;
    
    /**
     * <pre>
     *  Created by fanjiajia on 2019/3/29.
     *  desc:
     */
    
    public class DataModel {
    
        public void login(String userName, String Password, iCallBack<ResultBean> callBack) {
    
            ResultBean bean; // 封装返回的bean
    
            // 登录验证
            if (userName.equals("张三") && Password.equals("123456")) {
    
               bean = new ResultBean("200", "登录成功", "");
    
                callBack.callSuccess(bean);
            }else {
                bean = new ResultBean("110", "登录失败", "");
                callBack.callFailure(bean);
            }
        }
    }
    

    这里有个iCallback,故名思义,是结果的回调。看看他的定义;

    package com.mypractisedemos.module.mvpdemo;
    
    /**
     * <pre>
     *  Created by fanjiajia on 2019/3/29.
     *  desc:
     */
    
    public interface iCallBack<ResultBean>{
    
        void callError(ResultBean bean);
    
        void callSuccess(ResultBean bean);
    
        void callFailure(ResultBean bean);
    
        void callException(ResultBean bean);
    }
    

    这里有四种情况,成功,失败,异常,和错误。每一个回调都需要有个ResultBean,来看看这个回调对象的定义;

    package com.mypractisedemos.module.mvpdemo;
    
    /**
     * <pre>
     *  Created by fanjiajia on 2019/3/29.
     *  desc: 回调对象
     */
    
    public class ResultBean {
    
        private String code;    // 状态码
    
        private String msg; // 消息
    
        private String data;    // 数据 json格式
    
        public String getCode() {
            return code == null ? "" : code;
        }
    
        public void setCode(String code) {
            this.code = code;
        }
    
        public String getMsg() {
            return msg == null ? "" : msg;
        }
    
        public void setMsg(String msg) {
            this.msg = msg;
        }
    
        public String getData() {
            return data == null ? "" : data;
        }
    
        public void setData(String data) {
            this.data = data;
        }
    
        public ResultBean(String code, String msg, String data) {
            this.code = code;
            this.msg = msg;
            this.data = data;
        }
    
        @Override
        public String toString() {
            return "ResultBean{" +
                    "code='" + code + '\'' +
                    ", msg='" + msg + '\'' +
                    ", data='" + data + '\'' +
                    '}';
        }
    }
    
    

    这个参考了之前学长封装的一个OkHttp3工具的回调对象,用在这里了。

    iView

    看了Model层,现在看看试图接口定义一个接口,如下:

    package com.mypractisedemos.module.mvpdemo;
    
    /**
     * <pre>
     *  Created by fanjiajia on 2019/3/29.
     *  desc:
     */
    
    public interface iView {
    
        void loginSuccess(String resultMsg); // 登录成功回调
    
        void loginFailure(String failureMsg);// 登录失败回调
    }
    

    这里很简单,主要是登录活动的回调。

    DataPresenter

    然后开看看Presenter,之前说了这是model和view的桥梁,当点击登录按钮之后,肯定会调用其中的相应方法。先看定义;

    package com.mypractisedemos.module.mvpdemo;
    
    /**
     * <pre>
     *  Created by fanjiajia on 2019/3/29.
     *  desc:
     */
    
    public class DataPresenter {
    
        private iView mView;// 视图View
    
            private DataModel mModel;// 数据Model
    
            public DataPresenter(iView view) { // 构造 
            this.mView = view;
            this.mModel = new DataModel();
        }
    
        /**
         * 定义View会发起的动作
         */
    
        public void login(String userName, String password) {
    
            mModel.login(userName, password, new iCallBack<ResultBean>() {
                @Override
                public void callError(ResultBean resultBean) {
    
                }
    
                @Override
                public void callSuccess(ResultBean resultBean) {
                    mView.loginSuccess(resultBean.getMsg());
                }
    
                @Override
                public void callFailure(ResultBean resultBean) {
                    mView.loginFailure(resultBean.getMsg());
                }
    
                @Override
                public void callException(ResultBean resultBean) {
    
                }
            });
        }
    }
    

    这里看到他有两个属性,一个是View,另一个是Model,这就是为什么他是二者的桥梁。这里面的login方法中直接调用model的登录方法,然后通过iCallBack回调。

    Activity

    最后来看看我们之前操作比较多的View

    package com.mypractisedemos.module.mvpdemo;
    
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.view.View;
    import android.widget.Button;
    import android.widget.EditText;
    import android.widget.Toast;
    
    import com.jiajia.mypractisedemos.R;
    
    import butterknife.BindView;
    import butterknife.ButterKnife;
    
    public class LoginMvpActivity extends AppCompatActivity implements iView{ // 实现iView接口
    
        @BindView(R.id.et_mvp_login_username)
        EditText et_username; // 用户名
        @BindView(R.id.et_mvp_login_password)
        EditText et_password;   // 密码
        @BindView(R.id.btn_mvp_login_login)
        Button btn_login; // 登录按钮
    
        private DataPresenter mPresenter; // 声明Presenter
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_login_mvp);
    
            ButterKnife.bind(this); // ButterKnift框架的使用
    
            mPresenter = new DataPresenter(this); // 绑定自己的Presenter
    
        }
    
        public void login(View view) { // 登录按钮的onClick事件
    
            mPresenter.login(String.valueOf(et_username.getText()), String.valueOf(et_password.getText()));
        }
    
        @Override
        public void loginSuccess(String resultMsg) {
            Toast.makeText(this, resultMsg, Toast.LENGTH_SHORT).show();
        }
    
        @Override
        public void loginFailure(String failureMsg) {
            Toast.makeText(this, failureMsg, Toast.LENGTH_SHORT).show();
    
        }
    }
    
    

    这里我们的activity实现iView接口,并实现相应的方法即可!

    分析

    最后看看我的结构,果然是小白,因为把所有的东西都放在这个包下,没有层次感,因为是第一次尝试,只关注了实现思路本身。后面再进行分类。

    结构
    这里没有做内存泄漏的处理,来看看为何会导致内存泄漏,在activity中我们mPresenter = new DataPresenter(this); // 绑定自己的Presenter,注意,这里的this,然后在DataPresenter中this.mView = view;,这就说明了,Presenter持有了View的引用,而View(Activity)如果被切换,导致销毁,但是Presenter持有其强引用,导致无法被GC回收,因此导致这个Activity泄漏。下一篇再谈;
    GitHub: MVP-初体验(一)

    最后

    小白拙见!
    此致,敬礼!

    相关文章

      网友评论

          本文标题:小白用Android MVP-初体验(一)

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