MVP开发模式及简单架构封装

MVP开发模式

MVP图解.png
  • Model: 主要用于业务操作,如:网络请求,数据存储等
  • Presenter: 主要用于逻辑处理,沟通 MV ,尽可能不包含Android的代码
  • View: 主要用于规定界面的行为

优点

  • 由于Presenter层的出现,减少了View的逻辑操作和负担。这样使 View 与 Model 之间耦合度低
  • 合理规划的话,模块分明,模块复用率高,便于测试

缺点

  • 由于抽出了一层 Presenter,所以导致类和代码有一定的增加
  • 如果不能进行合理规划的将会导致后期模块杂乱,代码冗余度高
  • 纠结将 服务广播 放在何处
  • Presenter 会持有 View 的引用,如果不进行解绑会造成内存泄露

说到最后,其实 MVP 只是一种思想,没有什么固定的代码,固定的格式。因为多在实践中,慢慢理解和多多总结。不过在开发前一定要做好项目分析规划,切忌立马动手写代码。

MVP架构封装使用(登录功能示范)

GitHub传送门 --->MVPzzz(包含以下示例)

MVPzzz架构封装(未完成)

导包

JitPack

Step 1.
allprojects {
    repositories {
        ...
        maven { url 'https://jitpack.io' }
    }
}
  • 在所有的 repositories 中都加入上面语句,否则无法导入成功
Step 2.
dependencies {
    compile 'com.github.KittoZZZ:MVPzzz:v.0.0.2'
}

包含契约类

目录结构.png

1.建立View和Presenter的契约类

public class LoginContract {
    public interface ILoginView extends IBaseView {
        void LoginSuccess();

        void loginFail(String msg);
    }

    public interface ILoginPresenter {
        void toLogin(User user);
    }
}
  • View层接口必须继承IBaseView

  • 可以很明显的看出 View 和 Presenter 的关系

2.新建Model类(结合RxJava)

public class UserModel extends BaseModel {
    public Observable<String> toLogin(User user) {
        //Retrofit2
        String result = "fail";
        if ("zzz".equals(user.getAccount()) && "123".equals(user.getPassword())) {
            result = "success";
        }
        return Observable.just(result);
    }
}
  • BaseModel中并为没有实现什么功能,只是先留出,后序可能进行一些更改
  • 只是用于模拟所以并没有进行网络请求,写死数据

3.新建Presenter类实现契约类中的P层接口

    public class LoginPresenter extends BasePresenter<LoginContract.ILoginView> implements ILoginContract.ILoginPresenter {
    @InjectModel
    private UserModel userModel;

    @Override
    public void toLogin(User user) {
        userModel.toLogin(user)
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(String result) throws Exception {
                        if ("success".equals(result)) {
                            getView().LoginSuccess();
                        } else {
                            getView().loginFail(result);
                        }
                    }
                });
    }
}
  • getView() 方法用于调用 View 层的接口方法,如:LoginSuccess(),loginFail()
  • 必须要写BasePresenter泛型,LoginContract.ILoginView,否则无法调用它的方法
  • 使用注解 @InjectModel 进行注入 Model 对象

4.新建Activity类实现契约类中的V层接口

public class LoginActivity extends BaseMvpActivity implements LoginContract.ILoginView {
    private Button btnLogin;
    private EditText etAccount;
    private EditText etPassword;

    @InjectPresenter
    private LoginPresenter loginPresenter;

    @Override
    protected int setContentView() {
        return R.layout.activity_login;
    }

    @Override
    protected void initView() {
        etAccount = this.findViewById(R.id.et_account);
        etPassword = this.findViewById(R.id.et_password);
        btnLogin = this.findViewById(R.id.btn_login);
        btnLogin.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                String account = etAccount.getText().toString().trim();
                String password = etPassword.getText().toString().trim();
                User user = new User(account, password);
                loginPresenter.toLogin(user);
            }
        });
    }

    @Override
    protected void initData() {

    }

    @Override
    public void LoginSuccess() {
        Intent intent = new Intent(this, MainActivity.class);
        startActivity(intent);
        finish();
    }

    @Override
    public void loginFail(String msg) {
        Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
    }
}
  • 继承 BaseMvpActivity 实现 LoginContract.ILoginView ,重写方法
  • 使用注解 @InjectPresenter 进行注入 Presenter 对象

去除契约类(多个Presenter例子)

"根据设计原则 第一条 单一原则 LoginContract 这个类就把v和p耦合了 应该把view接口 和 presenter分离开 作为两个独立接口 然后分别由view和presenter子类实现"

上面是来自大牛的评论,经过思考确实将 V 层和 P 层的耦合了,所以思考将契约类去除的做法。

假设,在登录界面中的记住密码功能,需要在登录成功后将账号和密码进行保存在本地。下次登录的时候,直接读取显示在对应的输入框中。

目录结构改.png
  • 从目录结构中可以看出多了 UserDataPresenterDataModel 两个关键的类
  • 主要是 V 层在初始化的时候调用 UserDataPresenter 去读取,登录成功的时候保存数据
  • 由于只是个例子所以 DataModel 使用 SP 进行数据的存储
  • AppContext 只是用于在 M 层获取 ApplicationContext 的工具类

LoginActivity部分代码

public class LoginActivity extends BaseMvpActivity implements ILoginView, IUserDataView {
    ...
    
    @InjectPresenter
    private LoginPresenter loginPresenter;
    @InjectPresenter
    private UserDataPresenter userDataPresenter;

    ...

    @Override
    protected void initData() {
        btnLogin.setClickable(false);
        userDataPresenter.loadLastData();
    }

    @Override
    public void showLoginLoading() {
        Log.i("login", "showLoginLoading: 正在登陆");
    }

    @Override
    public void hideLoginLoading() {
        Log.i("login", "hideLoginLoading: 登录结束");
    }

    @Override
    public void loginSuccess() {
        Log.i("login", "loginSuccess: 登录成功");
        ...
        userDataPresenter.saveData(user);
    }

    @Override
    public void loginFail(String msg) {
        Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
    }

    @Override
    public void saveDataSuccess() {
        Log.i("login", "saveDataSuccess: 保存数据成功");
        ...
    }

    @Override
    public void saveDataFail(String msg) {
        Log.i("login", "saveDataFail: 保存数据失败 " + msg);
    }

    @Override
    public void readDataSuccess(User user) {
        Log.i("login", "readDataSuccess: 读取数据成功");
        btnLogin.setClickable(true);
        if (!TextUtils.isEmpty(user.getAccount()) && !TextUtils.isEmpty(user.getPassword())) {
            etAccount.setText(user.getAccount());
            etPassword.setText(user.getPassword());
           // loginPresenter.toLogin(user);
        }
    }

    @Override
    public void readDataFail(String msg) {
        Log.i("login", "loadDataFail: 读取数据失败" + msg);
    }
}
  • 在界面打开的时候开始读取保存本地的数据,此时登录按钮为不可点击
  • 读取成功,将登录按钮设为可点击,并且将数据现在输入框中
  • 如果自动登录的话,如果满足条件(账号密码不为空)则直接调用 P 层去登录

UserDataPresenter代码

public class UserDataPresenter extends BasePresenter<IUserDataView> implements IUserDataPresenter {
    @InjectModel
    private DataModel dataModel;

    @Override
    public void saveData(User user) {
        dataModel.saveUserData(user)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(String result) throws Exception {
                        if ("success".equals(result)) {
                            getView().saveDataSuccess();
                        } else {
                            getView().saveDataFail("保存失败");
                        }
                    }
                });
    }

    @Override
    public void loadLastData() {
        dataModel.readUserData()
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Consumer<User>() {
                    @Override
                    public void accept(User user) throws Exception {
                        getView().readDataSuccess(user);
                    }
                });
    }
}

DataModel代码就不展示,主要理解思想,详情可以看 --->MVPzzz(包含示例)


  • 上面展示的是多个 Presenter 情况使用注解来实例化,这种方式同样适用于多个 Model 的情况
  • 正是这个例子,让我觉得V层与P层是一一对应的形式存在,可以说是耦合的
  • 保留着契约类可以清晰的看出VP的关系,可能是个人水平较低,我还是会选择保留契约类的做法
  • 目前对于 MVP 开发模式的疑惑开始增多了.......
  • Fragment 的使用与 Activity 类似
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容