日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
Android中的MVP模式(帶實例)

最近在利用工作閑暇時間學(xué)習(xí)各種網(wǎng)絡(luò)的開源項目,也在搭建一個Android開源框架,希望能夠給對知識做一個總結(jié)。

創(chuàng)新互聯(lián)-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價比巴彥網(wǎng)站開發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫,直接使用。一站式巴彥網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋巴彥地區(qū)。費用合理售后完善,10多年實體公司更值得信賴。

這里利用一個簡單的應(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最不同的一點是M與V是不直接關(guān)聯(lián)的也是就Model與View不存在直接關(guān)系,這兩者之間間隔著的是Presenter層,其負責(zé)調(diào)控 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模式的應(yīng)用

2.1 model層描述和具體代碼

提供我們想要展示在view層的數(shù)據(jù)和具體登陸業(yè)務(wù)邏輯處理的實現(xiàn),

 
 
 
 
  1. package com.nsu.edu.androidmvpdemo.login; 
  2.  
  3.   
  4.  
  5. /** 
  6.  
  7. * Created by Anthony on 2016/2/15. 
  8.  
  9. * Class Note:模擬登陸的操作的接口,實現(xiàn)類為LoginModelImpl.相當于MVP模式中的Model層 
  10.  
  11. */ 
  12.  
  13. public interface LoginModel { 
  14.  
  15.     void login(String username, String password, OnLoginFinishedListener listener); 
  16.  
  17.  
  18.  
  19. package com.nsu.edu.androidmvpdemo.login; 
  20.  
  21.   
  22.  
  23. import android.os.Handler; 
  24.  
  25. import android.text.TextUtils; 
  26.  
  27. /** 
  28.  
  29. * Created by Anthony on 2016/2/15. 
  30.  
  31. * Class Note:延時模擬登陸(2s),如果名字或者密碼為空則登陸失敗,否則登陸成功 
  32.  
  33. */ 
  34.  
  35. public class LoginModelImpl implements LoginModel { 
  36.  
  37.   
  38.  
  39.     @Override 
  40.  
  41.     public void login(final String username, final String password, final OnLoginFinishedListener listener) { 
  42.  
  43.   
  44.  
  45.         new Handler().postDelayed(new Runnable() { 
  46.  
  47.             @Override public void run() { 
  48.  
  49.                 boolean error = false; 
  50.  
  51.                 if (TextUtils.isEmpty(username)){ 
  52.  
  53.                     listener.onUsernameError();//model層里面回調(diào)listener 
  54.  
  55.                     error = true; 
  56.  
  57.                 } 
  58.  
  59.                 if (TextUtils.isEmpty(password)){ 
  60.  
  61.                     listener.onPasswordError(); 
  62.  
  63.                     error = true; 
  64.  
  65.                 } 
  66.  
  67.                 if (!error){ 
  68.  
  69.                     listener.onSuccess(); 
  70.  
  71.                 } 
  72.  
  73.             } 
  74.  
  75.         }, 2000); 
  76.  
  77.     } 
  78.  
  79. }  

2.2 view層描述和具體代碼

負責(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點擊)

 
 
 
 
  1. package com.nsu.edu.androidmvpdemo.login; 
  2.  
  3.   
  4.  
  5. /** 
  6.  
  7. * Created by Anthony on 2016/2/15. 
  8.  
  9. * Class Note:登陸View的接口,實現(xiàn)類也就是登陸的activity 
  10.  
  11. */ 
  12.  
  13. public interface LoginView { 
  14.  
  15.     void showProgress(); 
  16.  
  17.   
  18.  
  19.     void hideProgress(); 
  20.  
  21.   
  22.  
  23.     void setUsernameError(); 
  24.  
  25.   
  26.  
  27.     void setPasswordError(); 
  28.  
  29.   
  30.  
  31.     void navigateToHome(); 
  32.  
  33.  
  34.  
  35. package com.nsu.edu.androidmvpdemo.login; 
  36.  
  37.   
  38.  
  39. import android.app.Activity; 
  40.  
  41. import android.content.Intent; 
  42.  
  43. import android.os.Bundle; 
  44.  
  45. import android.view.View; 
  46.  
  47. import android.widget.EditText; 
  48.  
  49. import android.widget.ProgressBar; 
  50.  
  51. import android.widget.Toast; 
  52.  
  53.   
  54.  
  55. import com.nsu.edu.androidmvpdemo.R; 
  56.  
  57.   
  58.  
  59. /** 
  60.  
  61. * Created by Anthony on 2016/2/15. 
  62.  
  63. * Class Note:MVP模式中View層對應(yīng)一個activity,這里是登陸的activity 
  64.  
  65. */ 
  66.  
  67. public class LoginActivity extends Activity implements LoginView, View.OnClickListener { 
  68.  
  69.   
  70.  
  71.     private ProgressBar progressBar; 
  72.  
  73.     private EditText username; 
  74.  
  75.     private EditText password; 
  76.  
  77.     private LoginPresenter presenter; 
  78.  
  79.   
  80.  
  81.     @Override 
  82.  
  83.     protected void onCreate(Bundle savedInstanceState) { 
  84.  
  85.         super.onCreate(savedInstanceState); 
  86.  
  87.         setContentView(R.layout.activity_login); 
  88.  
  89.   
  90.  
  91.         progressBar = (ProgressBar) findViewById(R.id.progress); 
  92.  
  93.         username = (EditText) findViewById(R.id.username); 
  94.  
  95.         password = (EditText) findViewById(R.id.password); 
  96.  
  97.         findViewById(R.id.button).setOnClickListener(this); 
  98.  
  99.   
  100.  
  101.         presenter = new LoginPresenterImpl(this); 
  102.  
  103.     } 
  104.  
  105.   
  106.  
  107.     @Override 
  108.  
  109.     protected void onDestroy() { 
  110.  
  111.         presenter.onDestroy(); 
  112.  
  113.         super.onDestroy(); 
  114.  
  115.     } 
  116.  
  117.   
  118.  
  119.     @Override 
  120.  
  121.     public void showProgress() { 
  122.  
  123.         progressBar.setVisibility(View.VISIBLE); 
  124.  
  125.     } 
  126.  
  127.   
  128.  
  129.     @Override 
  130.  
  131.     public void hideProgress() { 
  132.  
  133.         progressBar.setVisibility(View.GONE); 
  134.  
  135.     } 
  136.  
  137.   
  138.  
  139.     @Override 
  140.  
  141.     public void setUsernameError() { 
  142.  
  143.         username.setError(getString(R.string.username_error)); 
  144.  
  145.     } 
  146.  
  147.   
  148.  
  149.     @Override 
  150.  
  151.     public void setPasswordError() { 
  152.  
  153.         password.setError(getString(R.string.password_error)); 
  154.  
  155.     } 
  156.  
  157.   
  158.  
  159.     @Override 
  160.  
  161.     public void navigateToHome() { 
  162.  
  163. // TODO       startActivity(new Intent(this, MainActivity.class)); 
  164.  
  165.         Toast.makeText(this,"login success",Toast.LENGTH_SHORT).show(); 
  166.  
  167. //        finish(); 
  168.  
  169.     } 
  170.  
  171.   
  172.  
  173.     @Override 
  174.  
  175.     public void onClick(View v) { 
  176.  
  177.         presenter.validateCredentials(username.getText().toString(), password.getText().toString()); 
  178.  
  179.     } 
  180.  
  181.   
  182.  
  183. }  

2.3 presenter層描述和具體代碼 

Presenter扮演著view和model的中間層的角色。獲取model層的數(shù)據(jù)之后構(gòu)建view層;也可以收到view層UI上的反饋命令后分發(fā)處理邏輯,交給model層做業(yè)務(wù)操作。它也可以決定View層的各種操作。

 
 
 
 
  1. package com.nsu.edu.androidmvpdemo.login; 
  2.  
  3.   
  4.  
  5. /** 
  6.  
  7. * Created by Anthony on 2016/2/15. 
  8.  
  9. * Class Note:登陸的Presenter 的接口,實現(xiàn)類為LoginPresenterImpl,完成登陸的驗證,以及銷毀當前view 
  10.  
  11. */ 
  12.  
  13. public interface LoginPresenter { 
  14.  
  15.     void validateCredentials(String username, String password); 
  16.  
  17.   
  18.  
  19.     void onDestroy(); 
  20.  
  21.  
  22.  
  23. package com.nsu.edu.androidmvpdemo.login; 
  24.  
  25.   
  26.  
  27. /** 
  28.  
  29. * Created by Anthony on 2016/2/15. 
  30.  
  31. * Class Note: 
  32.  
  33. * 1 完成presenter的實現(xiàn)。這里面主要是Model層和View層的交互和操作。 
  34.  
  35. * 2  presenter里面還有個OnLoginFinishedListener, 
  36.  
  37. * 其在Presenter層實現(xiàn),給Model層回調(diào),更改View層的狀態(tài), 
  38.  
  39. * 確保 Model層不直接操作View層。如果沒有這一接口在LoginPresenterImpl實現(xiàn)的話, 
  40.  
  41. * LoginPresenterImpl只 有View和Model的引用那么Model怎么把結(jié)果告訴View呢? 
  42.  
  43. */ 
  44.  
  45. public class LoginPresenterImpl implements LoginPresenter, OnLoginFinishedListener { 
  46.  
  47.     private LoginView loginView; 
  48.  
  49.     private LoginModel loginModel; 
  50.  
  51.   
  52.  
  53.     public LoginPresenterImpl(LoginView loginView) { 
  54.  
  55.         this.loginView = loginView; 
  56.  
  57.         this.loginModel = new LoginModelImpl(); 
  58.  
  59.     } 
  60.  
  61.   
  62.  
  63.     @Override 
  64.  
  65.     public void validateCredentials(String username, String password) { 
  66.  
  67.         if (loginView != null) { 
  68.  
  69.             loginView.showProgress(); 
  70.  
  71.         } 
  72.  
  73.   
  74.  
  75.         loginModel.login(username, password, this); 
  76.  
  77.     } 
  78.  
  79.   
  80.  
  81.     @Override 
  82.  
  83.     public void onDestroy() { 
  84.  
  85.         loginView = null; 
  86.  
  87.     } 
  88.  
  89.   
  90.  
  91.     @Override 
  92.  
  93.     public void onUsernameError() { 
  94.  
  95.         if (loginView != null) { 
  96.  
  97.             loginView.setUsernameError(); 
  98.  
  99.             loginView.hideProgress(); 
  100.  
  101.         } 
  102.  
  103.     } 
  104.  
  105.   
  106.  
  107.     @Override 
  108.  
  109.     public void onPasswordError() { 
  110.  
  111.         if (loginView != null) { 
  112.  
  113.             loginView.setPasswordError(); 
  114.  
  115.             loginView.hideProgress(); 
  116.  
  117.         } 
  118.  
  119.     } 
  120.  
  121.   
  122.  
  123.     @Override 
  124.  
  125.     public void onSuccess() { 
  126.  
  127.         if (loginView != null) { 
  128.  
  129.             loginView.navigateToHome(); 
  130.  
  131.         } 
  132.  
  133.     } 
  134.  

2.4 登陸的回調(diào)接口

 
 
 
 
  1. package com.nsu.edu.androidmvpdemo.login; 
  2.  
  3.   
  4.  
  5. /** 
  6.  
  7. * Created by Anthony on 2016/2/15. 
  8.  
  9. * Class Note:登陸事件監(jiān)聽 
  10.  
  11. */ 
  12.  
  13. public interface OnLoginFinishedListener { 
  14.  
  15.   
  16.  
  17.     void onUsernameError(); 
  18.  
  19.   
  20.  
  21.     void onPasswordError(); 
  22.  
  23.   
  24.  
  25.     void onSuccess(); 
  26.  

 

demo的代碼流程:(請參考下面的類圖)

1 Activity做了一些UI初始化的東西并需要實例化對應(yīng)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處理完邏輯之后,結(jié)果通過OnLoginFinishedListener回調(diào)通知LoginPresenter

5 LoginPresenter再把結(jié)果返回給view層的Activity,***activity顯示結(jié)果

請參考這張類圖: 

本項目類圖

(3)注意:

3.1 presenter里面還有個OnLoginFinishedListener,其在Presenter層實現(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層可能是一個實現(xiàn)業(yè)務(wù)用例的交互者,在后續(xù)的文章中應(yīng)該會涉及到這方面的問題,目前能力有限。暫時講解到這里

(4)MVP經(jīng)典參考資料

請直接參考文章,這里面有很多的mvp模式的學(xué)習(xí)資料:

  • android架構(gòu)合集(請關(guān)注github,后續(xù)會不斷更新)
  • android mvp github地址(本篇博客正是參考這個項目進行講解的。這個項目也很簡單,分為login和main兩個模塊,總共十個類,思路非常清晰。學(xué)習(xí)的朋友可以直接clone查看源碼。) 

 

androidmvp 的src代碼分為login和main兩個模塊 

本項目為了簡單操作,只添加了login模塊

本項目github地址:

https://github.com/CameloeAnthony/AndroidMVPDemo


本文標題:Android中的MVP模式(帶實例)
本文路徑:http://www.5511xx.com/article/dghhgpo.html