Android中的MVP模式,帶實(shí)例

最近稍微了解了下MVP架構(gòu)模式,這篇文章寫得不錯,轉(zhuǎn)過來mark下:
原博客原地址:http://www.itdecent.cn/p/9d40b298eca9
項(xiàng)目github地址:https://github.com/CameloeAnthony/AndroidMVPDemo
最近在利用工作閑暇時間學(xué)習(xí)各種網(wǎng)絡(luò)的開源項(xiàng)目,也在搭建一個android開源框架,希望能夠給對知識做一個總結(jié)。這里利用一個簡單的應(yīng)用對MVP做一個講解。后面也有很多github源碼,都是特別經(jīng)典的例子,可以學(xué)習(xí)一下。
(1). MVP模式簡介
相信大家對MVC都是比較熟悉了:M-Model-模型、V-View-視圖、C-Controller-控制器,MVP作為MVC的演化版本,那么類似的MVP所對應(yīng)的意義:M-Model-模型、V-View-視圖、P-Presenter-表示器。 從MVC和MVP兩者結(jié)合來看,Controlller/Presenter在MVC/MVP中都起著邏輯控制處理的角色,起著控制各業(yè)務(wù)流程的作用。而 MVP與MVC最不同的一點(diǎn)是M與V是不直接關(guān)聯(lián)的也是就Model與View不存在直接關(guān)系,這兩者之間間隔著的是Presenter層,其負(fù)責(zé)調(diào)控 View與Model之間的間接交互。在 Android中很重要的一點(diǎn)就是對UI的操作基本上需要異步進(jìn)行也就是在MainThread中才能操作UI,所以對View與Model的切斷分離是 合理的。此外Presenter與View、Model的交互使用接口定義交互操作可以進(jìn)一步達(dá)到松耦合也可以通過接口更加方便地進(jìn)行單元測試。所以也就有了這張圖片(MVP和MVC的對比)

MVP和MVC的對比

其實(shí)最明顯的區(qū)別就是,MVC中是允許Model和View進(jìn)行交互的,而MVP中很明顯,Model與View之間的交互由Presenter完成。還有一點(diǎn)就是Presenter與View之間的交互是通過接口的(代碼中會體現(xiàn))。

(2). MVP模式的應(yīng)用
2.1 model層描述和具體代碼
提供我們想要展示在view層的數(shù)據(jù)和具體登陸業(yè)務(wù)邏輯處理的實(shí)現(xiàn),
package com.nsu.edu.androidmvpdemo.login;/** * Created by Anthony on 2016/2/15. * Class Note:模擬登陸的操作的接口,實(shí)現(xiàn)類為LoginModelImpl.相當(dāng)于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層里面回調(diào)listener error = true; } if (TextUtils.isEmpty(password)){ listener.onPasswordError(); error = true; } if (!error){ listener.onSuccess(); } } }, 2000); }}

2.2 view層描述和具體代碼
負(fù)責(zé)顯示數(shù)據(jù)、提供友好界面跟用戶交互就行。MVP下Activity和Fragment以及View的子類體現(xiàn)在了這一 層,Activity一般也就做加載UI視圖、設(shè)置監(jiān)聽再交由Presenter處理的一些工作,所以也就需要持有相應(yīng)Presenter的引用。本層所需要做的操作就是在每一次有相應(yīng)交互的時候,調(diào)用presenter的相關(guān)方法就行。(比如說,button點(diǎn)擊)
package com.nsu.edu.androidmvpdemo.login;/** * Created by Anthony on 2016/2/15. * Class Note:登陸View的接口,實(shí)現(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層對應(yīng)一個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ù)之后構(gòu)建view層;也可以收到view層UI上的反饋命令后分發(fā)處理邏輯,交給model層做業(yè)務(wù)操作。它也可以決定View層的各種操作。
package com.nsu.edu.androidmvpdemo.login;/** * Created by Anthony on 2016/2/15. * Class Note:登陸的Presenter 的接口,實(shí)現(xiàn)類為LoginPresenterImpl,完成登陸的驗(yàn)證,以及銷毀當(dāng)前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的實(shí)現(xiàn)。這里面主要是Model層和View層的交互和操作。 * 2 presenter里面還有個OnLoginFinishedListener, * 其在Presenter層實(shí)現(xiàn),給Model層回調(diào),更改View層的狀態(tài), * 確保 Model層不直接操作View層。如果沒有這一接口在LoginPresenterImpl實(shí)現(xiàn)的話, * LoginPresenterImpl只 有View和Model的引用那么Model怎么把結(jié)果告訴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 登陸的回調(diào)接口
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初始化的東西并需要實(shí)例化對應(yīng)LoginPresenter的引用和實(shí)現(xiàn) LoginView的接口,監(jiān)聽界面動作****2 登陸按鈕按下后即接收到登陸的事件,在onClick里接收到即通過LoginPresenter的引用把它交給LoginPresenter處理。LoginPresenter接收到了登陸的邏輯就知道要登陸了****3 然后LoginPresenter顯示進(jìn)度條并且把邏輯交給我們的Model去處理,也就是這里面的LoginModel,(LoginModel的實(shí)現(xiàn)類LoginModelImpl),同時會把OnLoginFinishedListener也就是LoginPresenter自身傳遞給我們的Model(LoginModel)。****4 LoginModel處理完邏輯之后,結(jié)果通過OnLoginFinishedListener回調(diào)通知LoginPresenter****5 LoginPresenter再把結(jié)果返回給view層的Activity,最后activity顯示結(jié)果請參考這張類圖:

本項(xiàng)目類圖

(3)注意:
3.1 presenter里面還有個OnLoginFinishedListener,其在Presenter層實(shí)現(xiàn),給Model層回調(diào),更改View層的狀態(tài),確保 Model層不直接操作View層。3.2 在一個好的架構(gòu)中,model層可能只是一個領(lǐng)域?qū)雍蜆I(yè)務(wù)邏輯層的入口,如果我們參考網(wǎng)上比較火的Uncle Bob clean architecture model層可能是一個實(shí)現(xiàn)業(yè)務(wù)用例的交互者,在后續(xù)的文章中應(yīng)該會涉及到這方面的問題,目前能力有限。暫時講解到這里

(4)MVP經(jīng)典參考資料
請直接參考文章,這里面有很多的mvp模式的學(xué)習(xí)資料:
android架構(gòu)合集(請關(guān)注github,后續(xù)會不斷更新)
android mvp github地址(本篇博客正是參考這個項(xiàng)目進(jìn)行講解的。這個項(xiàng)目也很簡單,分為login和main兩個模塊,總共十個類,思路非常清晰。學(xué)習(xí)的朋友可以直接clone查看源碼。)[圖片上傳中。。。(3)]androidmvp 的src代碼分為login和main兩個模塊

本項(xiàng)目為了簡單操作,只添加了login模塊

本項(xiàng)目github地址:https://github.com/CameloeAnthony/AndroidMVPDemo

作者:CameloeAnthony鏈接:http://www.itdecent.cn/p/9d40b298eca9來源:簡書著作權(quán)歸作者所有。商業(yè)轉(zhuǎn)載請聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請注明出處。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
禁止轉(zhuǎn)載,如需轉(zhuǎn)載請通過簡信或評論聯(lián)系作者。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容