關(guān)于MVP的介紹網(wǎng)上有很多湖笨,身為搬運(yùn)工的我直接搬來(lái)百度百科的旗扑,其實(shí)想來(lái)大家也都知道的。
MVP:
mvp的全稱(chēng)為Model-View-Presenter慈省,Model提供數(shù)據(jù)臀防,View負(fù)責(zé)顯示,Controller/Presenter負(fù)責(zé)邏輯的處理边败。
MVP與MVC有著一個(gè)重大的區(qū)別:在MVP中View并不直接使用Model袱衷,它們之間的通信是通過(guò)Presenter (MVC中的Controller)來(lái)進(jìn)行的,所有的交
互都發(fā)生在Presenter內(nèi)部笑窜,而在MVC中View會(huì)直接從Model中讀取數(shù)據(jù)而不是通過(guò) Controller致燥。
在學(xué)習(xí)MVP模式的時(shí)候,最近比較火的是谷歌的例子:
https://github.com/googlesamples/android-architecture
結(jié)構(gòu)很清晰怖侦,用一個(gè)契約類(lèi)將View篡悟,Presenter的接口寫(xiě)在里面谜叹,這樣可以清晰的知道有哪些功能,界面有哪些會(huì)有變化搬葬,數(shù)據(jù)有哪些需要處理荷腊。
在Presenter創(chuàng)建時(shí)傳入到View,同時(shí)將自身注入到View中急凰,甚是巧妙女仰。
來(lái)看一個(gè)小李子:
這是我模仿 掌上英雄聯(lián)盟做的界面 ,除去 ViewPager中的內(nèi)容抡锈,這個(gè)界面有兩個(gè)地方需要獲取數(shù)據(jù)疾忍。
一、輪播圖床三。
二一罩、Tab標(biāo)簽以及 “賽事”和“視頻” 這倆懸浮頁(yè)。
根據(jù)Fiddler聽(tīng)到的數(shù)據(jù)來(lái)看 撇簿,輪播圖是一個(gè)請(qǐng)求聂渊,Tab和懸浮頁(yè)是一個(gè)請(qǐng)求。
所以基本步驟就是 四瘫,定義設(shè)計(jì)Model汉嗽,BaseView ,BasePresenter找蜜。
這里的Model比較簡(jiǎn)單饼暑,看一下就過(guò)去了:
model接口:
public interface LoLItemModel {
Observable<LoLBean> getLoLNewsData(String id, int page, String plat, int version);
Observable<LoLBean> getLoLBanner( String plat, int version);
Observable<List<LoLTypeBean>> getLoLType(String plat, int version);
}
model實(shí)現(xiàn)類(lèi):
public class LoLItemModelImpl implements LoLItemModel {
@Override
public Observable<LoLBean> getLoLNewsData(String id, int page,String plat, int version) {
LoLService service = NetWork.getInstance().create(LoLService.class);
return service.getNewsLoL(id, page, plat, version);
}
@Override
public Observable<LoLBean> getLoLBanner(String plat, int version) {
LoLService service = NetWork.getInstance().create(LoLService.class);
return service.getLoLBanner(plat, version);
}
@Override
public Observable<List<LoLTypeBean>> getLoLType(String plat, int version) {
LoLService service = NetWork.getInstance().create(LoLService.class);
return service.getLoLType(plat, version);
}
}
Model 是在Presenter中進(jìn)行實(shí)例化操作的。
接著來(lái)看BaseView洗做,BasePresenter:
public interface IBaseView<T> {
void setPresenter(T presenter);
}
public interface IBasePresenter {
void start();
void destroy();
}
BasePresenter里面的方法看實(shí)際而定弓叛,我可能需要初始化一些東西,定義了start(),因?yàn)槭褂肦xJava竭望,要考慮生命周期邪码,所以定義一個(gè)destroy,在View結(jié)束時(shí)解綁咬清。
然后就是契約類(lèi)闭专,這個(gè)界面 需要和數(shù)據(jù)進(jìn)行交互的 只有設(shè)置輪播圖,tab頁(yè)旧烧。
所以就暫時(shí)先定下兩個(gè)方法影钉。
public class LoLMainContract {
public interface View extends IBaseView<Presenter> {
void initBanner(List<LoLBean.ListBean> bannerUrls);
void initTabs(List<LoLTypeBean> listbean);
}
public interface Presenter extends IBasePresenter {
void getBannersData();
void getTabsData();
}
}
實(shí)現(xiàn)Presenter:
在構(gòu)造函數(shù)中 獲取View同時(shí)將自身傳入 View中。
public class LoLMainPresenter implements LoLMainContract.Presenter {
private LoLMainContract.View mView;
LoLItemModelImpl lolIml;
Subscription getBannerSPN;
Subscription getTabSPN;
public LoLMainPresenter(LoLMainContract.View mView) {
this.mView = mView;
this.mView.setPresenter(this);
lolIml = new LoLItemModelImpl();
}
@Override
public void getBannersData() {
getBannerSPN = RxManager.getInstance().doSubscribe(lolIml.getLoLBanner("android", 9705), new Subscriber<LoLBean>() {
@Override
public void onCompleted() {
}
@Override
public void onError(Throwable e) {
}
@Override
public void onNext(LoLBean loLTypeBeen) {
mView.initBanner(loLTypeBeen.getList());
}
});
}
@Override
public void getTabsData() {
getTabSPN = RxManager.getInstance().doSubscribe(lolIml.getLoLType("android", 9705), new Subscriber<List<LoLTypeBean>>() {
@Override
public void onCompleted() {
}
@Override
public void onError(Throwable e) {
}
@Override
public void onNext(List<LoLTypeBean> loLTypeBeen) {
mView.initTabs(loLTypeBeen);
}
});
}
@Override
public void start() {
}
@Override
public void destroy() {
if (getBannerSPN != null)
getBannerSPN.unsubscribe();
if (getTabSPN != null)
getTabSPN.unsubscribe();
}
}
在View生命周期結(jié)束的時(shí)候?qū)τ^察者解綁掘剪。
然后就是實(shí)現(xiàn)View平委。
這里是Fragment 做View ,Activity統(tǒng)籌管理View
public class LoLMainFragment extends BaseFragment implements LoLMainContract.View{ }
我們的fragment實(shí)現(xiàn)契約類(lèi)中Presenter的View接口夺谁,
聲明Presenter并在實(shí)現(xiàn)View的方法中賦值:
private LoLMainContract.Presenter mPresenter;
@Override
public void setPresenter(LoLMainContract.Presenter presenter) {
this.mPresenter = presenter;
}
獲取數(shù)據(jù) :
mPresenter.getBannersData();
mPresenter.getTabsData();
獲取之后會(huì)調(diào)用View的方法 將返回的數(shù)據(jù) 在View中顯示廉赔。
都實(shí)現(xiàn)之后就是 將V,P實(shí)例出來(lái)肉微。
可以將方法放到 Fragment的 create中 。最好封裝起來(lái)
new LoLMainPresenter(this);
這樣一個(gè)基本的MVP模式就出來(lái)了蜡塌。
這只是練手用的碉纳,熟悉MVP構(gòu)建方法流程。但是我們不能為了架構(gòu)了使用架構(gòu)馏艾,視具體情況而定劳曹。
頁(yè)面功能多,邏輯復(fù)雜琅摩,那就得好好思量思量要怎么去實(shí)現(xiàn)能讓層次清晰铁孵,維護(hù)方便。
總之房资,開(kāi)心就好蜕劝。
最近學(xué)到的還有一種MVP實(shí)現(xiàn)方式,挺好的轰异,下篇來(lái)學(xué)習(xí)熙宇。
Mosby:https://github.com/sockeqwe/mosby