MVP,全称 Model-View-Presenter
优点:
1. 降低耦合度,实现了Model和View真正的完全分离,可以修改View而不影响Modle
2. 模块职责划分明显,层次清晰(下面会介绍Bob大叔的Clean Architecture)
3. 隐藏数据
4. Presenter可以复用,一个Presenter可以用于多个View,而不需要更改Presenter的逻辑(当然是在View的改动不影响业务逻辑的前提下)
5. 利于测试驱动开发。以前的Android开发是难以进行单元测试的(虽然很多Android开发者都没有写过测试用例,但是随着项目变得越来越复杂,没有测试是很难保证软件质量的;而且近几年来Android上的测试框架已经有了长足的发展——开始写测试用例吧),在使用MVP的项目中Presenter对View是通过接口进行,在对Presenter进行不依赖UI环境的单元测试的时候。可以通过Mock一个View对象,这个对象只需要实现了View的接口即可。然后依赖注入到Presenter中,单元测试的时候就可以完整的测试Presenter应用逻辑的正确性。
6. View可以进行组件化。在MVP当中,View不依赖Model。这样就可以让View从特定的业务场景中脱离出来,可以说View可以做到对业务完全无知。它只需要提供一系列接口提供给上层操作。这样就可以做到高度可复用的View组件。
7. 代码灵活性
缺点:
1. Presenter中除了应用逻辑以外,还有大量的View->Model,Model->View的手动同步逻辑,造成Presenter比较笨重,维护起来会比较困难。
2. 由于对视图的渲染放在了Presenter中,所以视图和Presenter的交互会过于频繁。
3. 如果Presenter过多地渲染了视图,往往会使得它与特定的视图的联系过于紧密。一旦视图需要变更,那么Presenter也需要变更了。
4. 额外的代码复杂度及学习成本。
小结:
(1) View :负责绘制UI元素、与用户进行交互(在Android中体现为Activity);
(2) View interface :需要View实现的接口,View通过View interface与Presenter进行交互,降低耦合,方便进行单元测试;
(3) Model :负责存储、检索、操纵数据(有时也实现一个Model interface用来降低耦合);
(4) Presenter :作为View与Model交互的中间纽带,处理与用户交互的负责逻辑。
代码示例:
先上 目录结构:
view代码:负责失败或者成功后的接口回调。
public interface LoginView {
void onSuccess(String msg);
}
model代码:负责数据的处理过程,比如请求http数据等。
public class LoginModel {
public boolean onLogin(String name, String pwd) {
boolean su = false;
if (name.equals("123456")) {
su = true;
} else {
su = false;
}
return su;
}
}
presenter代码:最重要的,连接view和model的纽带,负责两者之间的互通,将数据和UI 分离。
public class LoginPresenter {
private LoginModel model;
private LoginView view;
public LoginPresenter(LoginView view) {
this.view = view;
model = new LoginModel();
}
public void onClicked(String name, String pwd) {
if (model.onLogin(name, pwd)) {
view.onSuccess("success");
} else {
view.onSuccess("error");
}
}
}
最后就是activity代码:只负责UI显示,不负责数据处理过程。
public class MainActivity extends AppCompatActivity implements LoginView {
private EditText tvname, tvpwd;
private Button btn;
private LoginPresenter presenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tvname = findViewById(R.id.edit_name);
tvpwd = findViewById(R.id.edit_pwd);
btn = findViewById(R.id.btn);
presenter = new LoginPresenter(this);
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
presenter.onClicked(tvname.getText().toString(), tvpwd.getText().toString());
}
});
}
@Override
public void onSuccess(String msg) {
Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
}
}
1.0版本完成。