MVP簡介
???????隨著時(shí)代的發(fā)展别渔,移動(dòng)端UI越來越豐富、業(yè)務(wù)越來越復(fù)雜匾灶,用戶對界面的美觀棱烂、渲染的速度、數(shù)據(jù)讀取的速度要求也越來越高阶女。為了在保證擁有酷炫的UI颊糜、復(fù)雜的業(yè)務(wù)邏輯的條件下給用戶更好的體驗(yàn),需要對view秃踩、model衬鱼、controller職責(zé)進(jìn)行細(xì)分,讓view專注于處理數(shù)據(jù)的可視化以及和用戶的交互憔杨、讓model專注于數(shù)據(jù)的處理鸟赫。一種新的模式MVP(Model - View - Presenter)應(yīng)運(yùn)而生。
MVP和MVC的差異
???????MVP模式是從MVC模式理念的基礎(chǔ)上衍生出來的消别,在MVC的基礎(chǔ)上對V和M進(jìn)行了絕對的解耦抛蚤,兩者之間只有通過Presenter才能通信,優(yōu)化了V層和C層寻狂,職責(zé)劃分更明確岁经、單一。MVC中蛇券,M層和V層是可以通信的蒿偎,MVP中M層和V層是完全隔離的,只能通過P層來通信怀读,如圖(這里借圖一用)。
???????在MVC模式時(shí)期骑脱,由于UI越來越豐富菜枷、邏輯處理越來越復(fù)雜,Activity中對UI的活動(dòng)展示以及和用戶的交互也越來越復(fù)雜叁丧,代碼量比較大啤誊,同時(shí)受限于Android的線程安全以及很多時(shí)候需要使用到Activity的生命周期岳瞭,我們有不少的Controller層的代碼也會(huì)寫在Activity里面,也就是Activity同時(shí)承擔(dān)了View層和部分Controller層的工作蚊锹,影響開發(fā)效率瞳筏,不利于維護(hù)和擴(kuò)展。
???????于是牡昆,我們將復(fù)雜的邏輯移至另外一個(gè)類Presenter姚炕,讓Activity只負(fù)責(zé)UI的展示以及和用戶的交互。View層丢烘,創(chuàng)建View Interface柱宦,定義好View層的活動(dòng)事件,Activity加載布局播瞳、實(shí)現(xiàn)View Interface掸刊,將View Interface傳入Presenter實(shí)例,通過View Innterface和Presenter實(shí)現(xiàn)交互赢乓;Model層忧侧,負(fù)責(zé)存儲(chǔ)、檢索牌芋、操縱數(shù)據(jù)蚓炬,可以通過創(chuàng)建一個(gè)Model Interface來進(jìn)一步實(shí)現(xiàn)解耦;Presenter層姜贡,作為View和Model交互的中間樞紐试吁,負(fù)責(zé)接收用戶操作,調(diào)用Model層楼咳,將數(shù)據(jù)反饋到View層展示給用戶熄捍。
選擇MVP的必要性
???????1)降低了耦合性;
???????2)各模塊職責(zé)劃分明確母怜,利于協(xié)同開發(fā)余耽;
???????3)業(yè)務(wù)邏輯獨(dú)立出來,通過接口調(diào)用苹熏,方便單元測試碟贾;
???????4)代碼重用性高;
示例
???????這里跟隨很多大牛的腳步轨域,同樣使用登錄頁面來作為第一次體驗(yàn)MVP的示例袱耽。
???????①登錄頁面布局activity_main.xml代碼,布局比較簡單干发,兩個(gè)輸入框朱巨,一個(gè)登錄按鈕,一個(gè)加載等待條(默認(rèn)隱藏)枉长。
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".LoginActivity"
android:padding="20dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:layout_centerVertical="true"
>
<EditText
android:id="@+id/userName"
android:layout_width="match_parent"
android:layout_height="50dp"
android:hint="用戶名"/>
<EditText
android:id="@+id/password"
android:layout_width="match_parent"
android:layout_height="50dp"
android:layout_marginTop="5dp"
android:hint="密碼"/>
<Button
android:id="@+id/loginBtn"
android:layout_width="match_parent"
android:layout_height="60dp"
android:layout_marginTop="10dp"
android:text="Login"/>
</LinearLayout>
<ProgressBar
android:id="@+id/progressBar"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:visibility="gone"/>
</RelativeLayout>
???????②View Interface冀续,這里定義的是View的活動(dòng)琼讽,這里登錄頁面存在的活動(dòng)有(顯示加載等待框、隱藏加載等待框洪唐、返回登錄成功或者登錄失斪甑拧),所以ILoginView代碼如下凭需。
public interface ILoginView {
/*顯示等待框*/
void showProgress();
/*隱藏等待框*/
void hideProgress();
/*登錄成功*/
void loginSuccess();
/*登錄失敗*/
void loginError();
}
???????③Model Interface问欠,需要在model中去判斷用戶名和密碼是否正確,并返回處理結(jié)果功炮,所以ILoginModel需要定義一個(gè)checkLogin方法溅潜,再定義一個(gè)回調(diào)的內(nèi)部接口,因?yàn)橹挥蠵能主動(dòng)通信M薪伏,M不能主動(dòng)通信P滚澜,所有這里采用回調(diào)來返回結(jié)果。
public interface ILoginModel {
/*回調(diào)接口*/
interface OnLoginListener{
/*登錄成功回調(diào)*/
void loginSuccess();
/*登錄失敗回調(diào)*/
void loginError();
}
/*判斷輸入的用戶名嫁怀、密碼设捐,調(diào)用內(nèi)部接口回調(diào)*/
void checkLogin(String userName,String password,OnLoginListener listener);
}
???????④實(shí)現(xiàn)ILoginModel,這里虛擬登錄塘淑。
public class LoginModelImpl implements ILoginModel {
@Override
public void checkLogin(final String userName, final String password,
final OnLoginListener listener) {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
if(userName.equals("1") && password.equals("1")){
listener.loginSuccess();
}else{
listener.loginError();
}
}
},3000);
}
}
???????⑤Presenter Interface萝招,Presenter接收用戶點(diǎn)擊按鈕的事件,然后調(diào)用Model處理數(shù)據(jù)存捺,ILoginPresenter需要toLogin方法槐沼。
public interface ILoginPresenter {
void toLogin(String userName,String password);
}
???????⑥實(shí)現(xiàn)Presenter,P層作為V和M的交互媒介捌治,需要同時(shí)持有V和M岗钩,在M處理完返回的時(shí)候還需要通知到V層,所以還要實(shí)現(xiàn)M層的回調(diào)接口肖油。由于V層可以主動(dòng)溝通P層兼吓,所以V層和P層互相持有,這里V層直接在構(gòu)造函數(shù)里傳遞過來森枪。
public class LoginPresenterImpl implements ILoginPresenter,
ILoginModel.OnLoginListener {
private ILoginView mView;
private ILoginModel mModel;
public LoginPresenterImpl(ILoginView iLoginView) {
mView = iLoginView;
mModel = new LoginModelImpl();
}
@Override
public void toLogin(String userName, String password) {
mView.showProgress();
mModel.checkLogin(userName,password,this);
}
@Override
public void loginSuccess() {
mView.hideProgress();
mView.loginSuccess();
}
@Override
public void loginError() {
mView.hideProgress();
mView.loginError();
}
}
???????⑦View實(shí)現(xiàn)加載视搏,這里在Acitvity中操作,實(shí)現(xiàn)View Interface县袱,實(shí)例化Presenter浑娜,將View Interface作為參數(shù)傳遞給Presenter。
package liu.wolf.firstmvpdemo;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ProgressBar;
import android.widget.Toast;
import liu.wolf.firstmvpdemo.presenter.LoginPresenterImpl;
import liu.wolf.firstmvpdemo.view.ILoginView;
public class LoginActivity extends AppCompatActivity implements
ILoginView{
private ProgressBar progressBar;
private EditText userName,password;
private Button loginBtn;
private LoginPresenterImpl mPresenter = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mPresenter = new LoginPresenterImpl(this);
progressBar = findViewById(R.id.progressBar);
userName = findViewById(R.id.userName);
password = findViewById(R.id.password);
loginBtn = findViewById(R.id.loginBtn);
loginBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mPresenter.toLogin(userName.getText().toString(),
password.getText().toString());
}
});
}
@Override
public void showProgress() {
progressBar.setVisibility(View.VISIBLE);
}
@Override
public void hideProgress() {
progressBar.setVisibility(View.GONE);
}
@Override
public void loginSuccess() {
Toast.makeText(LoginActivity.this,"登錄成功式散!",
Toast.LENGTH_LONG).show();
}
@Override
public void loginError() {
Toast.makeText(LoginActivity.this,"登錄失斉锓摺!",
Toast.LENGTH_LONG).show();
}
}
代碼下載:https://download.csdn.net/download/liujibin1836591303/10374061
閱讀原文