最近稍微了解了下MVP架構(gòu)模式埋心,這篇文章寫得不錯(cuò)指郁,轉(zhuǎn)過來mark下:
原博客原地址:http://www.reibang.com/p/9d40b298eca9
項(xiàng)目github地址:https://github.com/CameloeAnthony/AndroidMVPDemo
最近在利用工作閑暇時(shí)間學(xué)習(xí)各種網(wǎng)絡(luò)的開源項(xiàng)目,也在搭建一個(gè)android開源框架拷呆,希望能夠給對知識(shí)做一個(gè)總結(jié)闲坎。這里利用一個(gè)簡單的應(yīng)用對MVP做一個(gè)講解疫粥。后面也有很多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的對比)
其實(shí)最明顯的區(qū)別就是,MVC中是允許Model和View進(jìn)行交互的须蜗,而MVP中很明顯硅确,Model與View之間的交互由Presenter完成。還有一點(diǎn)就是Presenter與View之間的交互是通過接口的(代碼中會(huì)體現(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:延時(shí)模擬登陸(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)交互的時(shí)候,調(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)一個(gè)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里面還有個(gè)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)聽界面動(dòng)作****2 登陸按鈕按下后即接收到登陸的事件讯蒲,在onClick里接收到即通過LoginPresenter的引用把它交給LoginPresenter處理捐迫。LoginPresenter接收到了登陸的邏輯就知道要登陸了****3 然后LoginPresenter顯示進(jìn)度條并且把邏輯交給我們的Model去處理,也就是這里面的LoginModel爱葵,(LoginModel的實(shí)現(xiàn)類LoginModelImpl)施戴,同時(shí)會(huì)把OnLoginFinishedListener也就是LoginPresenter自身傳遞給我們的Model(LoginModel)。****4 LoginModel處理完邏輯之后萌丈,結(jié)果通過OnLoginFinishedListener回調(diào)通知LoginPresenter****5 LoginPresenter再把結(jié)果返回給view層的Activity赞哗,最后activity顯示結(jié)果請參考這張類圖:
(3)注意:
3.1 presenter里面還有個(gè)OnLoginFinishedListener,其在Presenter層實(shí)現(xiàn)辆雾,給Model層回調(diào)肪笋,更改View層的狀態(tài),確保 Model層不直接操作View層度迂。3.2 在一個(gè)好的架構(gòu)中藤乙,model層可能只是一個(gè)領(lǐng)域?qū)雍蜆I(yè)務(wù)邏輯層的入口,如果我們參考網(wǎng)上比較火的Uncle Bob clean architecture model層可能是一個(gè)實(shí)現(xiàn)業(yè)務(wù)用例的交互者惭墓,在后續(xù)的文章中應(yīng)該會(huì)涉及到這方面的問題坛梁,目前能力有限。暫時(shí)講解到這里
(4)MVP經(jīng)典參考資料
請直接參考文章腊凶,這里面有很多的mvp模式的學(xué)習(xí)資料:
android架構(gòu)合集(請關(guān)注github划咐,后續(xù)會(huì)不斷更新)
android mvp github地址(本篇博客正是參考這個(gè)項(xiàng)目進(jìn)行講解的拴念。這個(gè)項(xiàng)目也很簡單,分為login和main兩個(gè)模塊褐缠,總共十個(gè)類政鼠,思路非常清晰。學(xué)習(xí)的朋友可以直接clone查看源碼队魏。)[圖片上傳中公般。。胡桨。(3)]androidmvp 的src代碼分為login和main兩個(gè)模塊
本項(xiàng)目為了簡單操作,只添加了login模塊
本項(xiàng)目github地址:https://github.com/CameloeAnthony/AndroidMVPDemo
作者:CameloeAnthony鏈接:http://www.reibang.com/p/9d40b298eca9來源:簡書著作權(quán)歸作者所有官帘。商業(yè)轉(zhuǎn)載請聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請注明出處登失。