Android MVP 的簡(jiǎn)單介紹與使用

MVP模式已經(jīng)火了很長(zhǎng)時(shí)間,似乎已經(jīng)火的過(guò)時(shí)奕枝。雖然現(xiàn)在MVVM風(fēng)頭正勁,但是MVP模式使用簡(jiǎn)單方便瓶堕,并使得耦合度降低隘道。對(duì)于新手開發(fā)APP,整體把握項(xiàng)目架構(gòu)是個(gè)很好的選擇郎笆。今天谭梗,我作為一個(gè)同樣的新手,和大家一起學(xué)習(xí)下MVP模式宛蚓。

一激捏、MVP概述

MVP,全稱 Model-View-Presenter凄吏,即模型-視圖-層現(xiàn)器远舅。

提到MVP,就必須要先介紹一下它的前輩MVC痕钢,因?yàn)镸VP正是基于MVC的基礎(chǔ)發(fā)展而來(lái)的图柏。兩個(gè)之間的關(guān)系也是源遠(yuǎn)流長(zhǎng)。

MVC盖喷,全稱Model-View-Controller爆办,即模型-視圖-控制器。具體如下:

View:對(duì)應(yīng)于布局文件
Model:業(yè)務(wù)邏輯和實(shí)體模型
Controllor:對(duì)應(yīng)于Activity

但是View對(duì)應(yīng)于布局文件课梳,其實(shí)能做的事情特別少距辆,實(shí)際上關(guān)于該布局文件中的數(shù)據(jù)綁定的操作余佃,事件處理的代碼都在Activity中,造成了Activity既像View又像Controller跨算,使得Activity變得臃腫爆土。

而當(dāng)將架構(gòu)改為MVP以后,Presenter的出現(xiàn)诸蚕,將Actvity視為View層步势,Presenter負(fù)責(zé)完成View層與Model層的交互。現(xiàn)在是這樣的:

View 對(duì)應(yīng)于Activity背犯,負(fù)責(zé)View的繪制以及與用戶交互
Model 依然是業(yè)務(wù)邏輯和實(shí)體模型
Presenter 負(fù)責(zé)完成View于Model間的交互

下面兩幅圖通過(guò)數(shù)據(jù)與視圖之間的交互清楚地展示了這種變化:



MVC模式下實(shí)際上就是Activty與Model之間交互坏瘩,View完全獨(dú)立出來(lái)了。



MVP模式通過(guò)Presenter實(shí)現(xiàn)數(shù)據(jù)和視圖之間的交互漠魏,簡(jiǎn)化了Activity的職責(zé)倔矾。同時(shí)即避免了View和Model的直接聯(lián)系,又通過(guò)Presenter實(shí)現(xiàn)兩者之間的溝通柱锹。

總結(jié):MVP模式減少了Activity的職責(zé)哪自,簡(jiǎn)化了Activity中的代碼,將復(fù)雜的邏輯代碼提取到了Presenter中進(jìn)行處理禁熏,模塊職責(zé)劃分明顯壤巷,層次清晰。與之對(duì)應(yīng)的好處就是瞧毙,耦合度更低胧华,更方便的進(jìn)行測(cè)試。

MVC和MVP的區(qū)別


MVC中是允許Model和View進(jìn)行交互的升筏,而MVP中很明顯撑柔,Model與View之間的交互由Presenter完成。還有一點(diǎn)就是Presenter與View之間的交互是通過(guò)接口的您访。

還有一點(diǎn)注意:MVC中V對(duì)應(yīng)的是布局文件铅忿,MVP中V對(duì)應(yīng)的是Activity。

二灵汪、MVP的簡(jiǎn)單使用

大多數(shù)MVP模式的示例都使用登錄案例進(jìn)行介紹檀训。因?yàn)楹?jiǎn)單方便,同時(shí)能提現(xiàn)出MVP的特點(diǎn)享言。今天我們也以此例進(jìn)行學(xué)習(xí)峻凫。
使用MVP的好處之一就是模塊職責(zé)劃分明顯,層次清晰览露。
該例的結(jié)構(gòu)圖即可展現(xiàn)此優(yōu)點(diǎn)荧琼。


1.Model層

在本例中,M0del層負(fù)責(zé)對(duì)從登錄頁(yè)面獲取地帳號(hào)密碼進(jìn)行驗(yàn)證(一般需要請(qǐng)求服務(wù)器進(jìn)行驗(yàn)證,本例直接模擬這一過(guò)程)命锄。
從上圖的包結(jié)構(gòu)圖中可以看出堰乔,Model層包含內(nèi)容:

①實(shí)體類bean
②接口,表示Model層所要執(zhí)行的業(yè)務(wù)邏輯
③接口實(shí)現(xiàn)類脐恩,具體實(shí)現(xiàn)業(yè)務(wù)邏輯镐侯,包含的一些主要方法

下面以代碼的形式一一展開。
①實(shí)體類bean

public class User {
    private String password;
    private String username;

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    @Override
    public String toString() {
        return "User{" +
                "password='" + password + '\'' +
                ", username='" + username + '\'' +
                '}';
    }
}

封裝了用戶名驶冒、密碼苟翻,方便數(shù)據(jù)傳遞。
②接口

public interface LoginModel {
    void login(User user, OnLoginFinishedListener listener);
}

其中OnLoginFinishedListener 是presenter層的接口骗污,方便實(shí)現(xiàn)回調(diào)presenter崇猫,通知presenter業(yè)務(wù)邏輯的返回結(jié)果,具體在presenter層介紹需忿。
③接口實(shí)現(xiàn)類

public class LoginModelImpl implements LoginModel {
    @Override
    public void login(User user, final OnLoginFinishedListener listener) {
        final String username = user.getUsername();
        final String password = user.getPassword();
        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);
    }
}

實(shí)現(xiàn)Model層邏輯:延時(shí)模擬登陸(2s)邓尤,如果用戶名或者密碼為空則登陸失敗,否則登陸成功贴谎。

2.View層

視圖:將Modle層請(qǐng)求的數(shù)據(jù)呈現(xiàn)給用戶。一般的視圖都只是包含用戶界面(UI)季稳,而不包含界面邏輯擅这,界面邏輯由Presenter來(lái)實(shí)現(xiàn)。

從上圖的包結(jié)構(gòu)圖中可以看出景鼠,View包含內(nèi)容:
①接口仲翎,上面我們說(shuō)過(guò)Presenter與View交互是通過(guò)接口。其中接口中方法的定義是根據(jù)Activity用戶交互需要展示的控件確定的铛漓。
②接口實(shí)現(xiàn)類溯香,將上述定義的接口中的方法在Activity中對(duì)應(yīng)實(shí)現(xiàn)具體操作。

下面以代碼的形式一一展開浓恶。
①接口

public interface LoginView {
    //login是個(gè)耗時(shí)操作玫坛,我們需要給用戶一個(gè)友好的提示,一般就是操作ProgressBar
    void showProgress();

    void hideProgress();
   //login當(dāng)然存在登錄成功與失敗的處理包晰,失敗給出提示
    void setUsernameError();

    void setPasswordError();
   //login成功湿镀,也給個(gè)提示
    void showSuccess();
}

上述5個(gè)方法都是presenter根據(jù)model層返回結(jié)果需要view執(zhí)行的對(duì)應(yīng)的操作。
②接口實(shí)現(xiàn)類
即對(duì)應(yīng)的登錄的Activity伐憾,需要實(shí)現(xiàn)LoginView接口勉痴。

public class LoginActivity extends AppCompatActivity 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);
       //創(chuàng)建一個(gè)presenter對(duì)象,當(dāng)點(diǎn)擊登錄按鈕時(shí)树肃,讓presenter去調(diào)用model層的login()方法蒸矛,驗(yàn)證帳號(hào)密碼
        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 showSuccess() {
         progressBar.setVisibility(View.GONE);
        Toast.makeText(this,"login success",Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onClick(View v) {
        User user = new User();
        user.setPassword(password.getText().toString());
        user.setUsername(username.getText().toString());
        presenter.validateCredentials(user);
    }

}

View層實(shí)現(xiàn)Presenter層需要調(diào)用的控件操作,方便Presenter層根據(jù)Model層返回的結(jié)果進(jìn)行操作View層進(jìn)行對(duì)應(yīng)的顯示。

3.Presenter層

Presenter是用作Model和View之間交互的橋梁雏掠。
從上圖的包結(jié)構(gòu)圖中可以看出斩祭,Presenter包含內(nèi)容:
①接口,包含Presenter需要進(jìn)行Model和View之間交互邏輯的接口磁玉,以及上面提到的Model層數(shù)據(jù)請(qǐng)求完成后回調(diào)的接口停忿。
②接口實(shí)現(xiàn)類,即實(shí)現(xiàn)具體的Presenter類邏輯蚊伞。

下面以代碼的形式一一展開席赂。
①接口

public interface OnLoginFinishedListener {
    void onUsernameError();

    void onPasswordError();

    void onSuccess();
}

當(dāng)Model層得到請(qǐng)求的結(jié)果,需要回調(diào)Presenter層时迫,讓Presenter層調(diào)用View層的接口方法颅停。

public interface LoginPresenter {
    void validateCredentials(User user);

    void onDestroy();
}

登陸的Presenter 的接口,實(shí)現(xiàn)類為L(zhǎng)oginPresenterImpl掠拳,完成登陸的驗(yàn)證癞揉,以及銷毀當(dāng)前view。
②接口實(shí)現(xiàn)類

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(User user) {
        if (loginView != null) {
            loginView.showProgress();
        }

        loginModel.login(user, 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.showSuccess();
        }
    }
}

由于presenter完成二者的交互溺欧,那么肯定需要二者的實(shí)現(xiàn)類(通過(guò)傳入?yún)?shù)喊熟,或者new)。

presenter里面有個(gè)OnLoginFinishedListener姐刁, 其在Presenter層實(shí)現(xiàn)芥牌,給Model層回調(diào),更改View層的狀態(tài)聂使, 確保 Model層不直接操作View層壁拉。

示例展示:

代碼地址

三、總結(jié)

MVP模式的整個(gè)核心流程:

View與Model并不直接交互柏靶,而是使用Presenter作為View與Model之間的橋梁弃理。其中Presenter中同時(shí)持有View層的Interface的引用以及Model層的引用,而View層持有Presenter層引用屎蜓。當(dāng)View層某個(gè)界面需要展示某些數(shù)據(jù)的時(shí)候痘昌,首先會(huì)調(diào)用Presenter層的引用,然后Presenter層會(huì)調(diào)用Model層請(qǐng)求數(shù)據(jù)梆靖,當(dāng)Model層數(shù)據(jù)加載成功之后會(huì)調(diào)用Presenter層的回調(diào)方法通知Presenter層數(shù)據(jù)加載情況控汉,最后Presenter層再調(diào)用View層的接口將加載后的數(shù)據(jù)展示給用戶。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末返吻,一起剝皮案震驚了整個(gè)濱河市姑子,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌测僵,老刑警劉巖街佑,帶你破解...
    沈念sama閱讀 212,080評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件谢翎,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡沐旨,警方通過(guò)查閱死者的電腦和手機(jī)森逮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,422評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)磁携,“玉大人褒侧,你說(shuō)我怎么就攤上這事∫昶” “怎么了闷供?”我有些...
    開封第一講書人閱讀 157,630評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)统诺。 經(jīng)常有香客問(wèn)我歪脏,道長(zhǎng),這世上最難降的妖魔是什么粮呢? 我笑而不...
    開封第一講書人閱讀 56,554評(píng)論 1 284
  • 正文 為了忘掉前任婿失,我火速辦了婚禮,結(jié)果婚禮上啄寡,老公的妹妹穿的比我還像新娘豪硅。我一直安慰自己,他們只是感情好挺物,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,662評(píng)論 6 386
  • 文/花漫 我一把揭開白布舟误。 她就那樣靜靜地躺著,像睡著了一般姻乓。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上眯牧,一...
    開封第一講書人閱讀 49,856評(píng)論 1 290
  • 那天蹋岩,我揣著相機(jī)與錄音,去河邊找鬼学少。 笑死剪个,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的版确。 我是一名探鬼主播扣囊,決...
    沈念sama閱讀 39,014評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼绒疗!你這毒婦竟也來(lái)了侵歇?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,752評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤吓蘑,失蹤者是張志新(化名)和其女友劉穎惕虑,沒(méi)想到半個(gè)月后坟冲,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,212評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡溃蔫,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,541評(píng)論 2 327
  • 正文 我和宋清朗相戀三年健提,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片伟叛。...
    茶點(diǎn)故事閱讀 38,687評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡私痹,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出统刮,到底是詐尸還是另有隱情紊遵,我是刑警寧澤,帶...
    沈念sama閱讀 34,347評(píng)論 4 331
  • 正文 年R本政府宣布网沾,位于F島的核電站癞蚕,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏辉哥。R本人自食惡果不足惜桦山,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,973評(píng)論 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望醋旦。 院中可真熱鬧恒水,春花似錦、人聲如沸饲齐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,777評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)捂人。三九已至御雕,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間滥搭,已是汗流浹背酸纲。 一陣腳步聲響...
    開封第一講書人閱讀 32,006評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留瑟匆,地道東北人闽坡。 一個(gè)月前我還...
    沈念sama閱讀 46,406評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像愁溜,于是被迫代替她去往敵國(guó)和親疾嗅。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,576評(píng)論 2 349

推薦閱讀更多精彩內(nèi)容