Java(Android)编程思想笔记03:在Android开发中使用MVP模式
1. MVP模式简介:
MVC模式相信大家肯定是比较熟悉的:M-Model-模型、V-View-视图、C-Controller-控制器。
MVP作为MVC的演化版本,那么类似的MVP所对应的意义:M-Model-模型、V-View-视图、P-Presenter-表示器。
从MVC和MVP两者结合来看:
Controlller/Presenter在MVC/MVP中都起着逻辑控制处理的角色,起着控制各业务流程的作用。
而 MVP与MVC最不同的一点是:Model与View是不直接关联的也是就Model与View不存在直接关系,这两者之间间隔着的是Presenter层,其负责调控 View与Model之间的间接交互。
在 Android中很重要的一点就是对UI的操作基本上需要异步进行也就是在MainThread中才能操作UI,所以对View与Model的切断分离是合理的。此外Presenter与View、Model的交互使用接口定义交互操作可以进一步达到松耦合也可以通过接口更加方便地进行单元测试。
所以也就有了这张图片(MVP和MVC的对比):
其实最明显的区别就是,MVC中是允许Model和View进行交互的,而MVP中很明显,Model与View之间的交互由Presenter完成。还有一点就是Presenter与View之间的交互是通过接口的(代码中会体现)。
因此,MVP具有以下优势:
(1)Model与View完全分离,我们可以修改View而不影响Model
(2)可以更高效地使用Model,因为所有的交互都发生在一个地方:Presenter内部
(3)我们可以将一个Presenter用于多个View,而不需要改变Presenter的逻辑。这个特性非常的有用,因为View的变化总是比Model的变化频繁。
(4)如果我们把逻辑放在Presenter中,那么我们就可以脱离用户接口来测试这些逻辑(单元测试)。
MVP的缺点:
(1)造成类数量爆炸,代码复杂度和学习成本高,在某些场景下presenter的复用会产生接口冗余。
2. MVP模式应用代码示例:
工程结构如下:
(1)Model层:
提供我们想要展示在view层的数据和具体登陆业务逻辑处理的实现:
1 package com.himi.mvpdemo.model; 2 3 import com.himi.mvpdemo.OnLoginFinishedListener; 4 5 /** 6 * Created by hebao on 2016/07/24. 7 * Class Note:模拟登陆的操作的接口,实现类为LoginModelImpl.相当于MVP模式中的Model层 8 */ 9 public interface LoginModel { 10 void login(String username, String password, OnLoginFinishedListener listener); 11 }
1 package com.himi.mvpdemo.model; 2 3 import com.himi.mvpdemo.OnLoginFinishedListener; 4 5 import android.os.Handler; 6 import android.text.TextUtils; 7 /** 8 * Created by hebao on 2016/7/24. 9 * Class Note:延时模拟登陆(2s),如果名字或者密码为空则登陆失败,否则登陆成功 10 */ 11 public class LoginModelImpl implements LoginModel { 12 13 @Override 14 public void login(final String username, final String password, final OnLoginFinishedListener listener) { 15 16 new Handler().postDelayed(new Runnable() { 17 @Override public void run() { 18 boolean error = false; 19 if (TextUtils.isEmpty(username)){ 20 listener.onUsernameError();//model层里面回调listener 21 error = true; 22 } 23 if (TextUtils.isEmpty(password)){ 24 listener.onPasswordError(); 25 error = true; 26 } 27 if (!error){ 28 listener.onSuccess(); 29 } 30 } 31 }, 2000); 32 } 33 }
(2)View层:
1 package com.himi.mvpdemo.view; 2 3 /** 4 * Created by hebao on 2016/7/24. 5 * Class Note:登陆View的接口,实现类也就是登陆的activity 6 */ 7 public interface LoginView { 8 void showProgress(); 9 10 void hideProgress(); 11 12 void setUsernameError(); 13 14 void setPasswordError(); 15 16 void navigateToHome(); 17 }
1 package com.himi.mvpdemo.view; 2 3 import com.himi.mvpdemo.presenter.LoginPresenter; 4 import com.himi.mvpdemo.presenter.LoginPresenterImpl; 5 6 import android.app.Activity; 7 import android.os.Bundle; 8 import android.view.View; 9 import android.widget.EditText; 10 import android.widget.ProgressBar; 11 import android.widget.Toast; 12 13 public class LoginActivity extends Activity implements LoginView, View.OnClickListener { 14 15 private ProgressBar progressBar; 16 private EditText username; 17 private EditText password; 18 private LoginPresenter presenter; 19 20 @Override 21 protected void onCreate(Bundle savedInstanceState) { 22 super.onCreate(savedInstanceState); 23 setContentView(R.layout.activity_login); 24 25 progressBar = (ProgressBar) findViewById(R.id.progress); 26 username = (EditText) findViewById(R.id.username); 27 password = (EditText) findViewById(R.id.password); 28 findViewById(R.id.button).setOnClickListener(this); 29 30 presenter = new LoginPresenterImpl(this); 31 } 32 33 @Override 34 protected void onDestroy() { 35 presenter.onDestroy(); 36 super.onDestroy(); 37 } 38 39 @Override 40 public void showProgress() { 41 progressBar.setVisibility(View.VISIBLE); 42 } 43 44 @Override 45 public void hideProgress() { 46 progressBar.setVisibility(View.GONE); 47 } 48 49 @Override 50 public void setUsernameError() { 51 username.setError(getString(R.string.username_error)); 52 } 53 54 @Override 55 public void setPasswordError() { 56 password.setError(getString(R.string.password_error)); 57 } 58 59 @Override 60 public void navigateToHome() { 61 // TODO startActivity(new Intent(this, MainActivity.class)); 62 Toast.makeText(this,"login success",Toast.LENGTH_SHORT).show(); 63 // finish(); 64 } 65 66 @Override 67 public void onClick(View v) { 68 presenter.validateCredentials(username.getText().toString(), password.getText().toString()); 69 } 70 71 }
(3)Presenter层:
Presenter扮演着view和model的中间层的角色。获取model层的数据之后构建view层;也可以收到view层UI上的反馈命令后分发处理逻辑,交给model层做业务操作。它也可以决定View层的各种操作。1 package com.himi.mvpdemo.presenter; 2 3 /** 4 * Created by hebao on 2016/7/24. Class Note:登陆的Presenter 5 * 的接口,实现类为LoginPresenterImpl,完成登陆的验证,以及销毁当前view 6 */ 7 public interface LoginPresenter { 8 void validateCredentials(String username, String password); 9 10 void onDestroy(); 11 }
1 package com.himi.mvpdemo.presenter; 2 3 import com.himi.mvpdemo.OnLoginFinishedListener; 4 import com.himi.mvpdemo.model.LoginModel; 5 import com.himi.mvpdemo.model.LoginModelImpl; 6 import com.himi.mvpdemo.view.LoginView; 7 8 public class LoginPresenterImpl implements LoginPresenter, OnLoginFinishedListener { 9 private LoginView loginView; 10 private LoginModel loginModel; 11 12 public LoginPresenterImpl(LoginView loginView) { 13 this.loginView = loginView; 14 this.loginModel = new LoginModelImpl(); 15 } 16 17 @Override 18 public void validateCredentials(String username, String password) { 19 if (loginView != null) { 20 loginView.showProgress(); 21 } 22 23 loginModel.login(username, password, this); 24 } 25 26 @Override 27 public void onDestroy() { 28 loginView = null; 29 } 30 31 @Override 32 public void onUsernameError() { 33 if (loginView != null) { 34 loginView.setUsernameError(); 35 loginView.hideProgress(); 36 } 37 } 38 39 @Override 40 public void onPasswordError() { 41 if (loginView != null) { 42 loginView.setPasswordError(); 43 loginView.hideProgress(); 44 } 45 } 46 47 @Override 48 public void onSuccess() { 49 if (loginView != null) { 50 loginView.navigateToHome(); 51 } 52 } 53 }
(4)登录的回调接口:
1 package com.himi.mvpdemo; 2 3 /** 4 * Created by hebao on 2016/7/24. 5 * Class Note:登陆事件监听 6 */ 7 public interface OnLoginFinishedListener { 8 9 void onUsernameError(); 10 11 void onPasswordError(); 12 13 void onSuccess(); 14 }
(5)部署程序到手机上,如下:
3. 示例代码流程(参考下面的图):
(1)Activity做了一些UI初始化的东西并需要实例化对应LoginPresenter的引用和实现 LoginView的接口,监听界面动作。
(2 )登陆按钮按下后即接收到登陆的事件,在onClick里接收到即通过LoginPresenter的引用把它交给LoginPresenter处理。LoginPresenter接收到了登陆的逻辑就知道要登陆了。
(3)然后LoginPresenter显示进度条并且把逻辑交给我们的Model去处理,也就是这里面的LoginModel,(LoginModel的实现类LoginModelImpl),同时会把OnLoginFinishedListener也就是LoginPresenter自身传递给我们的Model(LoginModel)。
(4)LoginModel处理完逻辑之后,结果通过OnLoginFinishedListener回调通知LoginPresenter。
(5)LoginPresenter再把结果返回给view层的Activity,最后activity显示结果。
请参考这张类图:
4. 注意事项:
(1)presenter里面还有个OnLoginFinishedListener,其在Presenter层实现,给Model层回调,更改View层的状态,确保 Model层不直接操作View层。
(2) 在一个好的架构中,model层可能只是一个领域层和业务逻辑层的入口,如果我们参考网上比较火的Uncle Bob clean architecture model层可能是一个实现业务用例的交互者,在后续的文章中应该会涉及到这方面的问题,目前能力有限。暂时讲解到这里.
5. MVP经典参考资料
请直接参考文章,这里面有很多的mvp模式的学习资料:
- android架构合集(请关注github,后续会不断更新)
- android mvp github地址(本篇博客正是参考这个项目进行讲解的。这个项目也很简单,分为login和main两个模块,总共十个类,思路非常清晰。学习的朋友可以直接clone查看源码。)
-
本项目Github地址:https://github.com/PocketBoy/hebao/tree/master/MVPDEMO