Android中的MVP模式(帶實例)
最近在利用工作閑暇時間學習各種網(wǎng)絡的開源項目,也在搭建一個android開源框架,希望能夠給對知識做一個總結。
這里利用一個簡單的應用對MVP做一個講解。后面也有很多github源碼,都是特別經(jīng)典的例子,可以學習一下。
(1). MVP模式簡介
相信大家對MVC都是比較熟悉了:M-Model-模型、V-View-視圖、C-Controller-控制器,MVP作為MVC的演化版本,那么類似的MVP所對應的意義:M-Model-模型、V-View-視圖、P-Presenter-表示器。 從MVC和MVP兩者結合來看,Controlller/Presenter在MVC/MVP中都起著邏輯控制處理的角色,起著控制各業(yè)務流程的作用。而 MVP與MVC最不同的一點是M與V是不直接關聯(lián)的也是就Model與View不存在直接關系,這兩者之間間隔著的是Presenter層,其負責調控 View與Model之間的間接交互。在 Android中很重要的一點就是對UI的操作基本上需要異步進行也就是在MainThread中才能操作UI,所以對View與Model的切斷分離是 合理的。此外Presenter與View、Model的交互使用接口定義交互操作可以進一步達到松耦合也可以通過接口更加方便地進行單元測試。所以也就有了這張圖片(MVP和MVC的對比)
MVP和MVC的對比
其實最明顯的區(qū)別就是,MVC中是允許Model和View進行交互的,而MVP中很明顯,Model與View之間的交互由Presenter完成。還有一點就是Presenter與View之間的交互是通過接口的(代碼中會體現(xiàn))。
(2). MVP模式的應用
2.1 model層描述和具體代碼
提供我們想要展示在view層的數(shù)據(jù)和具體登陸業(yè)務邏輯處理的實現(xiàn),
- package com.nsu.edu.androidmvpdemo.login;
- /**
- * Created by Anthony on 2016/2/15.
- * Class Note:模擬登陸的操作的接口,實現(xiàn)類為LoginModelImpl.相當于MVP模式中的Model層
- */
- public interface LoginModel {
- void login(String username, String password, OnLoginFinishedListener listener);
- }
- package com.nsu.edu.androidmvpdemo.login;
- import android.os.Handler;
- import android.text.TextUtils;
- /**
- * Created by Anthony on 2016/2/15.
- * Class Note:延時模擬登陸(2s),如果名字或者密碼為空則登陸失敗,否則登陸成功
- */
- public class LoginModelImpl implements LoginModel {
- @Override
- public void login(final String username, final String password, final OnLoginFinishedListener listener) {
- new Handler().postDelayed(new Runnable() {
- @Override public void run() {
- boolean error = false;
- if (TextUtils.isEmpty(username)){
- listener.onUsernameError();//model層里面回調listener
- error = true;
- }
- if (TextUtils.isEmpty(password)){
- listener.onPasswordError();
- error = true;
- }
- if (!error){
- listener.onSuccess();
- }
- }
- }, 2000);
- }
- }
2.2 view層描述和具體代碼
負責顯示數(shù)據(jù)、提供友好界面跟用戶交互就行。MVP下Activity和Fragment以及View的子類體現(xiàn)在了這一 層,Activity一般也就做加載UI視圖、設置監(jiān)聽再交由Presenter處理的一些工作,所以也就需要持有相應Presenter的引用。本層所需要做的操作就是在每一次有相應交互的時候,調用presenter的相關方法就行。(比如說,button點擊)
- package com.nsu.edu.androidmvpdemo.login;
- /**
- * Created by Anthony on 2016/2/15.
- * Class Note:登陸View的接口,實現(xiàn)類也就是登陸的activity
- */
- public interface LoginView {
- void showProgress();
- void hideProgress();
- void setUsernameError();
- void setPasswordError();
- void navigateToHome();
- }
- package com.nsu.edu.androidmvpdemo.login;
- import android.app.Activity;
- import android.content.Intent;
- import android.os.Bundle;
- import android.view.View;
- import android.widget.EditText;
- import android.widget.ProgressBar;
- import android.widget.Toast;
- import com.nsu.edu.androidmvpdemo.R;
- /**
- * Created by Anthony on 2016/2/15.
- * Class Note:MVP模式中View層對應一個activity,這里是登陸的activity
- */
- public class LoginActivity extends Activity implements LoginView, View.OnClickListener {
- private ProgressBar progressBar;
- private EditText username;
- private EditText password;
- private LoginPresenter presenter;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_login);
- progressBar = (ProgressBar) findViewById(R.id.progress);
- username = (EditText) findViewById(R.id.username);
- password = (EditText) findViewById(R.id.password);
- findViewById(R.id.button).setOnClickListener(this);
- presenter = new LoginPresenterImpl(this);
- }
- @Override
- protected void onDestroy() {
- presenter.onDestroy();
- super.onDestroy();
- }
- @Override
- public void showProgress() {
- progressBar.setVisibility(View.VISIBLE);
- }
- @Override
- public void hideProgress() {
- progressBar.setVisibility(View.GONE);
- }
- @Override
- public void setUsernameError() {
- username.setError(getString(R.string.username_error));
- }
- @Override
- public void setPasswordError() {
- password.setError(getString(R.string.password_error));
- }
- @Override
- public void navigateToHome() {
- // TODO startActivity(new Intent(this, MainActivity.class));
- Toast.makeText(this,"login success",Toast.LENGTH_SHORT).show();
- // finish();
- }
- @Override
- public void onClick(View v) {
- presenter.validateCredentials(username.getText().toString(), password.getText().toString());
- }
- }
2.3 presenter層描述和具體代碼
Presenter扮演著view和model的中間層的角色。獲取model層的數(shù)據(jù)之后構建view層;也可以收到view層UI上的反饋命令后分發(fā)處理邏輯,交給model層做業(yè)務操作。它也可以決定View層的各種操作。
- package com.nsu.edu.androidmvpdemo.login;
- /**
- * Created by Anthony on 2016/2/15.
- * Class Note:登陸的Presenter 的接口,實現(xiàn)類為LoginPresenterImpl,完成登陸的驗證,以及銷毀當前view
- */
- public interface LoginPresenter {
- void validateCredentials(String username, String password);
- void onDestroy();
- }
- package com.nsu.edu.androidmvpdemo.login;
- /**
- * Created by Anthony on 2016/2/15.
- * Class Note:
- * 1 完成presenter的實現(xiàn)。這里面主要是Model層和View層的交互和操作。
- * 2 presenter里面還有個OnLoginFinishedListener,
- * 其在Presenter層實現(xiàn),給Model層回調,更改View層的狀態(tài),
- * 確保 Model層不直接操作View層。如果沒有這一接口在LoginPresenterImpl實現(xiàn)的話,
- * LoginPresenterImpl只 有View和Model的引用那么Model怎么把結果告訴View呢?
- */
- public class LoginPresenterImpl implements LoginPresenter, OnLoginFinishedListener {
- private LoginView loginView;
- private LoginModel loginModel;
- public LoginPresenterImpl(LoginView loginView) {
- this.loginView = loginView;
- this.loginModel = new LoginModelImpl();
- }
- @Override
- public void validateCredentials(String username, String password) {
- if (loginView != null) {
- loginView.showProgress();
- }
- loginModel.login(username, password, this);
- }
- @Override
- public void onDestroy() {
- loginView = null;
- }
- @Override
- public void onUsernameError() {
- if (loginView != null) {
- loginView.setUsernameError();
- loginView.hideProgress();
- }
- }
- @Override
- public void onPasswordError() {
- if (loginView != null) {
- loginView.setPasswordError();
- loginView.hideProgress();
- }
- }
- @Override
- public void onSuccess() {
- if (loginView != null) {
- loginView.navigateToHome();
- }
- }
- }
2.4 登陸的回調接口
- package com.nsu.edu.androidmvpdemo.login;
- /**
- * Created by Anthony on 2016/2/15.
- * Class Note:登陸事件監(jiān)聽
- */
- public interface OnLoginFinishedListener {
- void onUsernameError();
- void onPasswordError();
- void onSuccess();
- }
demo的代碼流程:(請參考下面的類圖)
1 Activity做了一些UI初始化的東西并需要實例化對應LoginPresenter的引用和實現(xiàn) LoginView的接口,監(jiān)聽界面動作
2 登陸按鈕按下后即接收到登陸的事件,在onClick里接收到即通過LoginPresenter的引用把它交給LoginPresenter處理。LoginPresenter接收到了登陸的邏輯就知道要登陸了
3 然后LoginPresenter顯示進度條并且把邏輯交給我們的Model去處理,也就是這里面的LoginModel,(LoginModel的實現(xiàn)類LoginModelImpl),同時會把OnLoginFinishedListener也就是LoginPresenter自身傳遞給我們的Model(LoginModel)。
4 LoginModel處理完邏輯之后,結果通過OnLoginFinishedListener回調通知LoginPresenter
5 LoginPresenter再把結果返回給view層的Activity,***activity顯示結果
請參考這張類圖:
本項目類圖
(3)注意:
3.1 presenter里面還有個OnLoginFinishedListener,其在Presenter層實現(xiàn),給Model層回調,更改View層的狀態(tài),確保 Model層不直接操作View層。
3.2 在一個好的架構中,model層可能只是一個領域層和業(yè)務邏輯層的入口,如果我們參考網(wǎng)上比較火的Uncle Bob clean architecture model層可能是一個實現(xiàn)業(yè)務用例的交互者,在后續(xù)的文章中應該會涉及到這方面的問題,目前能力有限。暫時講解到這里
(4)MVP經(jīng)典參考資料
請直接參考文章,這里面有很多的mvp模式的學習資料:
- android架構合集(請關注github,后續(xù)會不斷更新)
- android mvp github地址(本篇博客正是參考這個項目進行講解的。這個項目也很簡單,分為login和main兩個模塊,總共十個類,思路非常清晰。學習的朋友可以直接clone查看源碼。)
androidmvp 的src代碼分為login和main兩個模塊
本項目為了簡單操作,只添加了login模塊
本項目github地址:
https://github.com/CameloeAnthony/AndroidMVPDemo