由于公司里的架构模式用到MVP,觉得自己还不够熟悉,决定在此理一理,并给大家一起总结下。
一 MVP模式介绍
MVP全称Model View Presenter。
MVP能够有效的降低View的复杂性,避免业务逻辑被塞进View中,防止View的代码变得冗杂。MVP模式会解除View与Model的耦合,同时又带来了良好的扩展性、可测试性,保证了系统的整洁性、灵活性。
肯能对于简单的应用来说MVP稍显麻烦,各种各样接口与概念,使得整个应用充斥着零散的接口,但是对于比较复杂的应用,MVP模式是一种良好的架构模式,它能够非常好的组织应用结构。
MVP模式可以分离显示层和逻辑层,它们之间通过接口进行通信,降低耦合。理想化的MVP可以实现同一份逻辑代码搭配不同的显示界面,因为它们之间并不依赖与具体,而是依赖与抽象。
其实这边讲View层和Model层分离,也是相比MVC模式有一个很大的改进,我们知道,在Android当中,我们对View的更改需要通过异步来出来,MVP模式切除了View与Model的联系,通过异步的方式来将二者联系在一起,这个分离做的很好。
对于一个可扩展、稳定的应用来说,我们需要定义分离各个层,主要是UI层、业务逻辑层和数据层。毕竟,我们不知道以后还要加入什么逻辑,是从本地数据库检索数据?还是从远程的服务器中?我们的UI、数据库是否会被替换?例如,随着产品的升级,我们的UI可能会被重新设计,若UI发生了变化,此时由于业务逻辑耦合在View中,UI变化导致我们修改新的View控件,此时你就需要到原来的View中抽离具体的业务逻辑,这将是一件令人非常折磨人也易于出错的事,到最终你还是要将业务逻辑抽离开来。
MVP并不是一个标准化的模式,它有很多种实现方式,我们也可以根据自己的需求和自己认为对的方式去修正MVP的实现方式,它可以随着Presenter的复杂层度变化。只要保证我们是通过Presenter将View和Model解耦合、降低类型复杂度、各个模块可以独立测试、独立变化,这就是正确的方向。
二、 MVP模式的三个角色
1.Presenter—–交互中间人
Presenter主要作为沟通View和Model的桥梁,它从Model层检索数据后,返回给View层,使得View和Model之间没有耦合,也将业务逻辑从View角色上抽离出来。
2.View—–用户界面
View通常是指Activity、Fragment或者某个View控件,它含有一个Presenter成员变量。通常View需要实现一个逻辑接口,将View上的操作转交给Presenter进行实现,最后,Presenter调用View逻辑接口将结果返回给View元素。
3.Model—–数据的存取
对于一个结构化的APP来说,Model角色主要是提供数据的存取功能。Presenter需要通过Model层存储、获取数据,Model就像一个数据仓库。更直白的说,Model是封装了数据库DAO或者网络获取数据的角色,或者两种数据获取方式的集合。
三、MVP实战举例
讲了那么多概念,这里我们以一个简单的登陆业务逻辑来应用我们的MVP模式。
3.1
首先,我们来看一下整体结构
bean层代表的是model层
presenter层代表是处理数据和View层的交互的
view层就是视图层
IUserBiz是一个接口,UserBiz是这个接口的实现类
3.2
首先看下model层
这个类比较简单,就是定义一些变量和方法
public class User {private String usename;private String password;public String getUsename() {return usename;}public void setUsename(String usename) {this.usename = usename;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}
}
接着看下IUserbiz,以及它的实现类IUserBiz,模拟登录操作,等登录操作做完,有一个回调接口来通知登录状态完成
public interface IUserBiz {public void login(String username,String password,OnLoginListener loginListener);
}
public class UserBiz implements IUserBiz {
@Overridepublic void login(final String username, final String password, final OnLoginListener loginListener) {//模拟子线程耗时操作Thread thread=new Thread(new Runnable() {@Overridepublic void run() {try{Thread.sleep(2000);}catch (InterruptedException e){e.printStackTrace();}//模拟登陆成功if("aaa".equals(username)&&"123".equals(password)){User user=new User();user.setUsename(username);user.setPassword(password);//回调接口loginListener.loginsuccess(user);}else {loginListener.loginFailed();}}});thread.start();}
}
接着看下View层,里面有很多操作的方法可以去重写,根据自己的经验和严谨的思维去设计这个接口
public interface IUserLoginView {String getUsername();String getPassword();void clearUserName();void clearPassword();void showLoading();void hideLoading();void toMainActivity(User user);void showFailedError();
}
我们继续看下这个的是实现类,也就是我们的activity,在这里实现了我们刚刚定义的方法因为我们用MVP的设计模式,使用的是我们的接口,使我们的代码看上去非常清晰
public class UserLoginActivity extends AppCompatActivity implements IUserLoginView {
...........@Overridepublic String getUsername() {return etUsername.getText().toString();}@Overridepublic String getPassword() {return etPassword.getText().toString();}@Overridepublic void clearUserName() {etUsername.setText("");}@Overridepublic void clearPassword() {etPassword.setText("");}@Overridepublic void showLoading() {mPbLoading.setVisibility(View.VISIBLE);}@Overridepublic void hideLoading() {mPbLoading.setVisibility(View.GONE);}@Overridepublic void toMainActivity(User user) {Toast.makeText(this,user.getUsename()+"login success,to MainActivity",Toast.LENGTH_LONG).show();}@Overridepublic void showFailedError() {Toast.makeText(this,"login failed",Toast.LENGTH_LONG).show();}
}
.....
}
最后,看下我们的Presenter,代表的就是数据和view交互的桥梁,方法比较简单,里面是具体的交互操作,逻辑非常清楚。
public class UserLoginPresenter {
...public void login(){userLoginView.showLoading();//实现接口userBiz.login(userLoginView.getUsername(), userLoginView.getPassword(), new OnLoginListener() {@Overridepublic void loginsuccess(final User user) {//需要在UI线程执行mHandler.post(new Runnable() {@Overridepublic void run() {userLoginView.toMainActivity(user);userLoginView.hideLoading();}});}@Overridepublic void loginFailed() {//需要在UI线程执行mHandler.post(new Runnable() {@Overridepublic void run() {userLoginView.showFailedError();userLoginView.hideLoading();}});}});}}
源码连接:
demo传送门