您好,登錄后才能下訂單哦!
今天就跟大家聊聊有關(guān)Android中MVP模式的作用是什么,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
(1). MVP模式簡(jiǎn)介
相信大家對(duì)MVC都是比較熟悉了:M-Model-模型、V-View-視圖、C-Controller-控制器,MVP作為MVC的演化版本,那么類似的MVP所對(duì)應(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)就是對(duì)UI的操作基本上需要異步進(jìn)行也就是在MainThread中才能操作UI,所以對(duì)View與Model的切斷分離是 合理的。此外Presenter與View、Model的交互使用接口定義交互操作可以進(jìn)一步達(dá)到松耦合也可以通過接口更加方便地進(jìn)行單元測(cè)試。所以也就有了這張圖片(MVP和MVC的對(duì)比)
MVP和MVC的對(duì)比
其實(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)類為L(zhǎng)oginModelImpl.相當(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層對(duì)應(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)類為L(zhǎng)oginPresenterImpl,完成登陸的驗(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的代碼流程:(請(qǐng)參考下面的類圖)
1 Activity做了一些UI初始化的東西并需要實(shí)例化對(duì)應(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é)果
請(qǐng)參考這張類圖:
本項(xiàng)目類圖
(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)典參考資料
請(qǐng)直接參考文章,這里面有很多的mvp模式的學(xué)習(xí)資料:
android架構(gòu)合集(請(qǐng)關(guān)注github,后續(xù)會(huì)不斷更新)
android mvp github地址(本篇博客正是參考這個(gè)項(xiàng)目進(jìn)行講解的。這個(gè)項(xiàng)目也很簡(jiǎn)單,分為login和main兩個(gè)模塊,總共十個(gè)類,思路非常清晰。學(xué)習(xí)的朋友可以直接clone查看源碼。)
androidmvp 的src代碼分為login和main兩個(gè)模塊
看完上述內(nèi)容,你們對(duì)Android中MVP模式的作用是什么有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。