1. 程式人生 > >Android---MVP模式介紹,以及MVP與MVC的區別

Android---MVP模式介紹,以及MVP與MVC的區別

(1). MVP模式簡介

相信大家對MVC都是比較熟悉了:M-Model-模型、V-View-檢視、C-Controller-控制器,MVP作為MVC的演化版本,那麼類似的MVP所對應的意義:M-Model-模型、V-View-檢視、P-Presenter-表示器。 從MVC和MVP兩者結合來看,Controlller/Presenter在MVC/MVP中都起著邏輯控制處理的角色,起著控制各業務流程的作用。而 MVP與MVC最不同的一點是M與V是不直接關聯的也是就Model與View不存在直接關係,這兩者之間間隔著的是Presenter層,其負責調控 View與Model之間的間接互動。在 Android中很重要的一點就是對UI的操作基本上需要非同步進行也就是在MainThread中才能操作UI,所以對View與Model的切斷分離是 合理的。此外Presenter與View、Model的互動使用介面定義互動操作可以進一步達到鬆耦合也可以通過介面更加方便地進行單元測試。所以也就有了這張圖片(MVP和MVC的對比)


MVP和MVC的對比

其實最明顯的區別就是,MVC中是允許Model和View進行互動的,而MVP中很明顯,Model與View之間的互動由Presenter完成。還有一點就是Presenter與View之間的互動是通過介面的(程式碼中會體現)。

(2). MVP模式的應用

2.1 model層描述和具體程式碼

提供我們想要展示在view層的資料和具體登陸業務邏輯處理的實現,

package com.nsu.edu.androidmvpdemo.login;

/**
 * Created by Anthony on 2016/2/15.
 * Class Note:模擬登陸的操作的介面,實現類為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層描述和具體程式碼

負責顯示資料、提供友好介面跟使用者互動就行。MVP下Activity和Fragment以及View的子類體現在了這一 層,Activity一般也就做載入UI檢視、設定監聽再交由Presenter處理的一些工作,所以也就需要持有相應Presenter的引用。本層所需要做的操作就是在每一次有相應互動的時候,呼叫presenter的相關方法就行。(比如說,button點選)

package com.nsu.edu.androidmvpdemo.login;

/**
 * Created by Anthony on 2016/2/15.
 * Class Note:登陸View的介面,實現類也就是登陸的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層的資料之後構建view層;也可以收到view層UI上的反饋命令後分發處理邏輯,交給model層做業務操作。它也可以決定View層的各種操作。

package com.nsu.edu.androidmvpdemo.login;

/**
 * Created by Anthony on 2016/2/15.
 * Class Note:登陸的Presenter 的介面,實現類為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的實現。這裡面主要是Model層和View層的互動和操作。
 * 2  presenter裡面還有個OnLoginFinishedListener,
 * 其在Presenter層實現,給Model層回撥,更改View層的狀態,
 * 確保 Model層不直接操作View層。如果沒有這一介面在LoginPresenterImpl實現的話,
 * 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:登陸事件監聽
 */
public interface OnLoginFinishedListener {

    void onUsernameError();

    void onPasswordError();

    void onSuccess();
}

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

1 Activity做了一些UI初始化的東西並需要例項化對應LoginPresenter的引用和實現 LoginView的介面,監聽介面動作
2 登陸按鈕按下後即接收到登陸的事件,在onClick裡接收到即通過LoginPresenter的引用把它交給LoginPresenter處理。LoginPresenter接收到了登陸的邏輯就知道要登陸了
3 然後LoginPresenter顯示進度條並且把邏輯交給我們的Model去處理,也就是這裡面的LoginModel,(LoginModel的實現類LoginModelImpl),同時會把OnLoginFinishedListener也就是LoginPresenter自身傳遞給我們的Model(LoginModel)。
4 LoginModel處理完邏輯之後,結果通過OnLoginFinishedListener回撥通知LoginPresenter
5 LoginPresenter再把結果返回給view層的Activity,最後activity顯示結果
請參考這張類圖:


本專案類圖

(3)注意:


3.1 presenter裡面還有個OnLoginFinishedListener,其在Presenter層實現,給Model層回撥,更改View層的狀態,確保 Model層不直接操作View層。

3.2 在一個好的架構中,model層可能只是一個領域層和業務邏輯層的入口,如果我們參考網上比較火的Uncle Bob clean architecture model層可能是一個實現業務用例的互動者

MVP與MVC區別

MVP

介紹

MVP模式(Model-View-Presenter)是MVC模式的一個衍生。主要目的是為了解耦,使專案易於維護。

  • Model 依然是業務邏輯和實體模型
  • View 經常由Activity實現,包含Presenter的引用。所要做的就是當有互動時,呼叫Presenter裡的對應方法。
  • Presenter 負責完成View於Model間的互動,從Model裡取資料,返回給View處理好的資料。

為什麼使用MVP

在以往的Android開發中,Activity並不是一個標準的MVC模式中的Controller, 它的載入應用的佈局和初始化使用者介面,接受並處理來自使用者的操作請求,進而作出響應。但是隨著介面及其邏輯的複雜度不斷提升,Activity類的職責不斷增加,以致變得龐大臃腫。當我們將其中複雜的邏輯處理移至另外的一個類(Presneter)中時,Activity其實就是MVP模式中View,它負責UI元素的初始化,建立UI元素與Presenter的關聯(Listener之類),同時自己也會處理一些簡單的邏輯(複雜的邏輯交由Presenter處理)。 
對於測試來說,在MVP模式中,處理複雜邏輯的Presenter是通過interface與View(Activity)進行互動的。我們可以通過自定義類實現這個interface來模擬Activity的行為對Presenter進行單元測試,省去了大量的部署及測試的時間。

MVC介紹

  • Model 是應用程式中用於處理應用程式資料邏輯的部分。
  • View 是應用程式中處理資料顯示的部分。
  • Controller是應用程式中處理使用者互動的部分。 

比較

MVP模式:

  • View不直接與Model互動 ,而是通過與Presenter互動來與Model間接互動
  • Presenter與View的互動是通過介面來進行的,更有利於新增單元測試
  • 通常View與Presenter是一對一的,但複雜的View可能繫結多個Presenter來處理邏輯

MVC模式:

  • View可以與Model直接互動
  • Controller是基於行為的,並且可以被多個View共享
  • 可以負責決定顯示哪個View

相關推薦

Android---MVP模式介紹以及MVPMVC區別

(1). MVP模式簡介 相信大家對MVC都是比較熟悉了:M-Model-模型、V-View-檢視、C-Controller-控制器,MVP作為MVC的演化版本,那麼類似的MVP所對應的意義:M-Model-模型、V-View-檢視、P-Presenter-表示器。 從MV

[android進階篇]MVP模式優化防止記憶體洩漏和空指標問題

MVPOptimize MVP模式優化 主要優化P層V層互相持有物件,不能及時回收/銷燬問題 如果你看過我的MVP整合教程【android進階篇】MVP+Retrofit+RxJava框架結合 你可能就會發現,如果頁面在請求的時候,網路不好,這時使用

Android MVPMVC區別

1、MVP與MVC的區別 Activity職責不同,Activity在MVP中是View層,在MVC中是Controller層,這是MVC和MVP很主要的一個區別,可以說Android從MVC轉向MVP開發也主要是優化Activity的程式碼,避免Activity的程式碼臃腫龐大。

多播委託觀察者模式聯合使用以及委託事件的區別

首先我們先看一下多播委託: 使用委託時,首先我們宣告委託,委託語法一共有四種類型,分別時有參,無參,以及有無引數返回值。 1 public class DelegateShow //: System.MulticastDelegate,多播委託 2 { 3 public del

android AppCompat, splash啟動白屏(黑屏)全屏去掉狀態列以及splash虛擬按鍵遮擋

在做專案是,需要加一個splash,以美化應用,消除app啟動等待的乏味 一、使用環境         使用Android Studio activity 繼承 AppCompatActivity, 實現splash功能         測試環境,沒有虛擬返回按鍵的安卓手機

Android新特性介紹ConstraintLayout完全解析

mat 區別 界面 -s 自己 解決 match roo pre 本篇文章的主題是ConstraintLayout。其實ConstraintLayout是Android Studio 2.2中主要的新增功能之一,也是Google在去年的I/O大會上重點宣傳的一個功能。我們都

android全平臺編譯ffmpeg以及x264fdk-aac實踐

目錄 編譯環境 單獨編譯完整功能ffmpeg庫 開始編譯x264庫 開始編譯fdk-aac庫 ffmpeg混合編譯x264和fdk-aar庫 編譯環境 作業系統:ubuntu 16.05 android-ndk-r10e 注意 這

plsql檢視是否鎖表模式以及解鎖SQL

--工作中的點滴積累 SELECT l.session_id sid, s.serial#, l.locked_mode 鎖模式, l.oracle_username 登入使用者, l.os_user_name 登入機器使用者名稱,

一覽Swift中的常用關鍵字 Swift - final關鍵字的介紹以及使用場景

要學習Swift這門語言,就必須先了解Swift的關鍵字及對應的解釋。這裡就列一下在Swift中常用到的關鍵字。 關鍵字是類似於識別符號的保留字元序列,除非用重音符號(`)將其括起來,否則不能用作識別符號。關鍵字是對編譯器具有特殊意義的預定義保留識別符號。常見的關鍵字有以下4種。  與

[C++] STL庫函式之字串string::npos的介紹以及string中的find函式~

npos經常和find一起用~它們兩個都在標頭檔案<string>裡面~先看用法: #include <iostream> #include <string> us

springboot的pom中找不到mysql-connector-java以及springbootspringcloud版本衝突

我springboot選用了2.0.5.RELEASE <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-sta

BLE Mesh 前言1:什麼是藍芽Mesh以及BLE之間的關係?

我們知道藍芽是目前普及率很廣的全球通用型無線通訊標準,他為智慧裝置帶來了簡捷,安全的連線。但是從它的第一版推出到5.0版本,還是主要針對星型拓撲網路。這也直接決定了它的網路覆蓋面積是有限的。 直到去年的藍芽Mesh標準的推出,才正式為藍芽的應用推向了更廣泛的空間。全新的藍芽

Docker入門實踐筆記(三)一篇文章搞懂Docker下安裝Redis以及RedisSpringBoot整合

@Configuration public class RedisConfig { ​ /** * 注入 RedisConnectionFactory */ @Autowired RedisConnectionFactory redisConnectionFacto

Redis詳細介紹以及在SpringBoot中的簡單使用

1.Redis的特點:   (1)Redis支援資料的持久化,可以將記憶體中的資料儲存到磁碟中,下次啟動的時候可以直接載入磁碟中的資料;   (2)Redis不僅可以儲存key-value型別的資料,還可以儲存List、Set、hash、zset等資料結構的資料;

Nexus 5 Android L 使用感受以及如何刷回 4.4 Kitkat

看了Google I/O後,對Material Design,電池省電模式等感到非常好奇,於是決定刷Android L 爽一把。 比較讓人討厭的是每次刷機後,之前裝的app的資料,都會被erase掉,還得重灌一遍。 好奇心趨勢吧,終於刷了Android L。刷機過程參考Google官方網

Android安全攻防戰反編譯混淆技術完全解析(上)

轉載請註明出處:http://blog.csdn.net/guolin_blog/article/details/49738023 之前一直有猶豫過要不要寫這篇文章,畢竟去反編譯人家的程式並不是什麼值得驕傲的事情。不過單純從技術角度上來講,掌握反編譯功能確實是

《連載 | 物聯網框架ServerSuperIO教程》- 14.配製工具介紹以及裝置驅動、檢視驅動、服務例項的掛載

目       錄 14.配製工具介紹,以及裝置驅動、檢視驅動、服務例項的掛載... 2 14.1        概述... 2 14.2        掛載裝置驅動... 2 14.3        掛載顯示檢視... 4 14.4 

kafka基本原理介紹以及重新選舉replica複製機制isr等。

最近做的專案,通過資料庫的log日誌將資料庫某些千萬量級的表(這些表需要聯表查詢)資料同步到elasticsearch中,以減輕資料庫的查詢壓力,其中以kafka作為訊息中介軟體,以下是做該專案過程中對kafka的一些整理。 一、中介軟體 中介軟體,用於業務對於資料的時效

【無監督學習】DBSCAN聚類演算法原理介紹以及程式碼實現

前言:無監督學習想快一點複習完,就轉入有監督學習 聚類演算法主要包括哪些演算法?主要包括:K-m

Android整合Google支付以及遇到的坑、坑

Google商店的應用被下架,應用內購買必須走Google支付,還要扣去百分之三十的手續費,而且有些國家還會收一定的銷售稅最高達27%,其實Google支付只是自己集成了Paypal支付和銀行卡支付,然後Google收手續費。使用者使用Google正常支付退款時間是48小時