MVP模式的作用
- 分離了視圖邏輯和業(yè)務邏輯,降低了耦合
- Activity只處理生命周期的任務珠移,代碼變得更加簡潔
- 視圖邏輯和業(yè)務邏輯分別抽象到了View和Presenter的接口中去,提高代碼的可閱讀性
- Presenter被抽象成接口暇韧,可以有多種具體的實現(xiàn)懈玻,所以方便進行單元測試
- 把業(yè)務邏輯抽到Presenter中去,避免后臺線程引用著Activity導致Activity的資源無法被系統(tǒng)回收+ 從而引起內(nèi)存泄露和OOM
其中最重要的有三點:
Activity 代碼變得更加簡潔
相信很多人閱讀代碼的時候涂乌,都是從Activity開始的湾盒,對著一個1000+行代碼的Activity,看了都覺得難受罚勾。
使用MVP之后尖殃,Activity就能瘦身許多了,基本上只有FindView场刑、SetListener以及Init的代碼蚪战。其他的就是對Presenter的調(diào)用,還有對View接口的實現(xiàn)邀桑。這種情形下閱讀代碼就容易多了壁畸,而且你只要看Presenter的接口,就能明白這個模塊都有哪些業(yè)務太抓,很快就能定位到具體代碼令杈。Activity變得容易看懂,容易維護掉丽,以后要調(diào)整業(yè)務、刪減功能也就變得簡單許多捶障。
MVP模式的使用
上面一張簡單的MVP模式的UML圖项炼,從圖中可以看出,使用MVP驱闷,至少需要經(jīng)歷以下步驟:
1空免、創(chuàng)建IPresenter接口,把所有業(yè)務邏輯的接口都放在這里扼菠,并創(chuàng)建它的實現(xiàn)PresenterCompl(在這里可以方便地查看業(yè)務功能坝咐,由于接口可以有多種實現(xiàn)所以也方便寫單元測試)
2、創(chuàng)建IView接口秧饮,把所有視圖邏輯的接口都放在這里泽篮,其實現(xiàn)類是當前的Activity/Fragment
3、由UML圖可以看出泼各,Activity里包含了一個IPresenter亏拉,而PresenterCompl里又包含了一個IView并且依賴了Model。Activity里只保留對IPresenter的調(diào)用莽使,其它工作全部留到PresenterCompl中實現(xiàn)
4磷蛹、Model并不是必須有的味咳,但是一定會有View和Presenter
什么是MVP架構
MVP就是Model-View-Presenter,MVP是從經(jīng)典的模式MVC演變而來责嚷,它們的基本思想有相通的地方:Controller/Presenter負責邏輯的處理掂铐,Model提供數(shù)據(jù),View負責顯示爆班。作為一種新的模式辱姨,MVP與MVC有著一個重大的區(qū)別:在MVP中View并不直接使用Model,它們之間的通信是通過Presenter (MVC中的Controller)來進行的枢舶,所有的交互都發(fā)生在Presenter內(nèi)部替久,而在MVC中View會直接從Model中讀取數(shù)據(jù)而不是通過 Controller蚯根。
MVP架構:
View: 對應于Activity,負責View的繪制以及與用戶交互
Model: 依然是業(yè)務邏輯和實體模型
Presenter: 負責完成View于Model間的交互
- View不直接與Model交互吼具,而是通過與Presenter交互來與Model間接交互矩距。
- Presenter與View的交互是通過接口來進行的。
- 通常View與Presenter是一對一的陡蝇,但復雜的View可能綁定多個Presenter來處理邏輯哮肚。
1允趟、Model層
首先是一個javabean User實體類
public class User {
private String name;
private String id;
private String sex;
private String age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public String getAge() {
return age;
}
public void setAge(String age) {
this.age = age;
}
}
Model層抽象接口實現(xiàn):
public class GetUserInfo implements IGetUser {
@Override
public void getUserInfo(final int id, final OnUserInfoListener listener) {
// 模擬子線程耗時操作
new Thread() {
@Override
public void run() {
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} // 模擬信息獲取成功
if (id == 1) {
User user = new User();
user.setName("非著名程序員");
user.setAge("26");
user.setSex("男");
user.setId("1");
listener.getUserInfoSuccess(user);
} else {
listener.getUserInfoFailed();
}
}
}.start();
}
}
Model層抽象接口
public interface IGetUser {
public void getUserInfo(int id, OnUserInfoListener listener);
}
public interface OnUserInfoListener {
void getUserInfoSuccess(User user);
void getUserInfoFailed();
}
2、View層
我們都知道Presenter與View交互是通過接口分唾,所以我們需要定義一個IShowUserView的接口狮斗,這個接口封裝的方法基本上都跟視圖展示有關。
public interface IShowUserView {
void showLoading();
void hideLoading();
void toMainActivity(User user);
void showFailedError();
}
3折砸、Presenter層
Presenter是Model和View之間交互的橋梁沙峻,里面有一些業(yè)務邏輯的操作摔寨。
public class UserInfoPresenter {
private IGetUser iGetUser;
private IShowUserView iShowUserView;
private Handler mHandler = new Handler();
public UserInfoPresenter(IShowUserView iShowUserView) {
this.iShowUserView = iShowUserView;
this.iGetUser = new GetUserInfo();
}
public void getUserInfoToShow(int id) {
iShowUserView.showLoading();
iGetUser.getUserInfo(id, new OnUserInfoListener() { @Override
public void getUserInfoSuccess(final User user) { // 需要在UI線程執(zhí)行
mHandler.post(new Runnable() { @Override
public void run() {
iShowUserView.toMainActivity(user);
iShowUserView.hideLoading();
}
});
} @Override
public void getUserInfoFailed() {
mHandler.post(new Runnable() { @Override
public void run() {
iShowUserView.showFailedError();
iShowUserView.hideLoading();
}
});
}
});
}
}
4祷肯、Activity中的調(diào)用
public class MainActivity extends Activity implements IShowUserView {
private Button btn;
private TextView name_tv, age_tv, sex_tv;
private ProgressDialog pd = null;
private UserInfoPresenter userInfoPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
userInfoPresenter = new UserInfoPresenter(this);
btn = (Button) findViewById(R.id.btn);
name_tv = (TextView) findViewById(R.id.name_tv);
age_tv = (TextView) findViewById(R.id.age_tv);
sex_tv = (TextView) findViewById(R.id.sex_tv);
pd = new ProgressDialog(this);
pd.setMessage("正在加載……");
btn.setOnClickListener(new View.OnClickListener() { @Override
public void onClick(View v) {
userInfoPresenter.getUserInfoToShow(1);
}
});
}
@Override
public void showLoading() {
pd.show();
}
@Override
public void hideLoading() {
pd.cancel();
}
@Override
public void toMainActivity(User user) {
name_tv.setText(user.getName());
age_tv.setText(user.getAge());
sex_tv.setText(user.getSex());
}
@Override
public void showFailedError() {
Toast.makeText(this, "獲取信息有誤", Toast.LENGTH_SHORT).show();
}
}