1.关于MVP架构
有时候在实现一个APP的时候会发现,写着写着Activity中的代码会越来越多,控件和数据的初始化没有问题。当添加按钮并绑定监听事件时候有网络访问,往往直接写在OnClick()方法中(我以前就是这么干的),Activity中的代码马上就变得十分冗余,当中出现了业务逻辑,数据访问部分,Activity简直变成了一个万能类。
Activity变成万能类的后果是,代码变得庞大并且臃肿,代码间严重耦合,改一个小部分往往牵一发而动全身,而且代码的可读性很差,想象一下在一个功能众多的Activity中寻找某一个功能某块的场景。
这就诞生了各种APP分层架构,其中MVP架构是最近十分流行的APP架构。
MVP的全称为:Model-View-Presenter
MVP将APP划分为三层,Model层负责数据存储与数据处理,View负责展示数据,Presenter层负责具体的业务逻辑。
虽然网上的对于MVP介绍的文章看了不少,但是MVC都还没搞明白,对于MVP就更是处于一知半解的状态,刚好Google官方在Github上推出了一个项目实例(点击查看),用于展示Android各种各样的MVP架构,其中的todo-mvp属于MVP基础架构实例,即使是基础架构的实现方案也使我受益匪浅,并仿照todo-mvp实现了一个小的案例。
2.案例介绍
在案例中使用了http://gank.io/api提供的福利接口,可从中获取各种妹子的福利照片并展示。
项目结构:
(1)BaseView和BasePresenter两个基类接口。
BaseView.java:
public interface BaseView<T> {
void setPresenter(T presenter);
}
BasePresenter.java:
public interface BasePresenter {
void start();
}
所有的View和Presenter首先实现这两个基类接口。
其中BaseView的setPresenter(T presenter)方法用于绑定View对应的Presenter,BasePresenter的start()方法用于初始化时执行相应的逻辑。
(2)具体的实现接口
之后在特定的功能中,在契约类WelfareDetailContact.java中声明具体的View和Presenter接口,在其中声明实现的具体功能。
WelfareDetailContact.java:
public interface WelfareDetailContact {
interface View extends BaseView<Presenter> {
void showPic(Welfare data);
}
interface Presenter extends BasePresenter {
void displayHome();
void refreshRandom();
}
}
其中View的showPic(Welfare data)方法只负责根据结构化的数据进行相应的显示,Presenter的两个方法只负责展示主页和随机刷新逻辑(目前只添加了这两个逻辑)。
具体的WelfareDetailView与WelfareDetailPresenter需实现契约类中的两个接口。
(3)Model数据层
WelfareDataSource.java:
public interface WelfareDataSource {
interface GetCallback {
void onWelfareGet(Welfare welfare);
void onDataNotAvailable();
}
void queryHome(GetCallback callback);
void queryPage(String page, GetCallback callback);
void queryRandom(GetCallback callback);
}
此接口声明了获取数据的方法和回调接口。
实现此接口的一共有三个类:WelfareRepository,WelfareLocalDataSource和WelfareRemoteDataSource。
WelfareRepository 负责总调度,WelfareLocalDataSource负责本地数据的加载与存储,WelfareRemoteDataSource负责网络远程数据的获取。
在本地和远程数据获取时,都会将JSON数据组装成Welfare对象后传入回调接口。
WelfareRepository 作为负责的总调度,分别持有WelfareLocalDataSource和WelfareRemoteDataSource的实例引用。三个类均使用单例模式。
下面看看具体实现:
总调度类WelfareRepository
WelfareRepository.java:
public class WelfareRepository implements WelfareDataSource {
private WelfareLocalDataSource mLocalDataSource;
private WelfareRemoteDataSource mRemoteDataSource;
......
@Override
public void queryHome(final GetCallback callback) {
mLocalDataSource.queryHome(new GetCallback() {
@Override
public void onWelfareGet(Welfare welfare) {
callback.onWelfareGet(welfare);
}
@Override
public void onDataNotAvailable() {
mRemoteDataSource.queryHome(new GetCallback() {
@Override
public void onWelfareGet(Welfare welfare) {
callback.onWelfareGet(welfare);
mLocalDataSource.updateWelfare(welfare);
}
@Override
public void onDataNotAvailable() {
}
});
}
});
}
......
}
以queryHome()查询主页为例,总调度WelfareRepository先是调用了WelfareLocalDataSource实例的queryHome()方法进行本地读取,若是本地读取成功,则正常执行传入的回调;若是本地数据获取失败,则调用WelfareRemoteDataSource实例的queryHome()方法获取网络数据请求,成功时正常执行回调,并在本地进行存储。
(4)具体View和Presenter的实现类。
WelfareDetailFragment.java:
public class WelfareDetailFragment extends Fragment implements WelfareDetailContact.View {
private WelfareDetailContact.Presenter mPresenter;
......
@Override
public void showPic(Welfare data) {
mAdapter.setData(data.results);
mAdapter.notifyDataSetChanged();
}
@Override
public void setPresenter(WelfareDetailContact.Presenter presenter) {
this.mPresenter = presenter;
}
@Override
public void onResume() {
super.onResume();
mPresenter.start();
}
}
在实例中使用Fragment来代替Activity来执行View的角色,这使得Activity中的代码更为简洁,使角色分工更加明确。
可以看到,WelfareDetailFragment作为View的角色,持有Presenter的引用,并在onResume()方法中调用mPresenter.start()来执行初始化时需要执行的逻辑。
WelfareDetailPresenter.java:
public class WelfareDetailPresenter implements WelfareDetailContact.Presenter {
private WelfareRepository mWelfareRepository;
private WelfareDetailContact.View mView;
......
public WelfareDetailPresenter(WelfareRepository welfareRepository, WelfareDetailContact.View view) {
this.mWelfareRepository = welfareRepository;
this.mView = view;
view.setPresenter(this);
}
@Override
public void start() {
displayHome();
}
@Override
public void displayHome() {
mWelfareRepository.queryHome(new WelfareDataSource.GetCallback() {
@Override
public void onWelfareGet(Welfare welfare) {
mView.showPic(welfare);
}
@Override
public void onDataNotAvailable() {
}
});
}
}
WelfareDetailPresenter在构造初始化时与相应的View进行绑定,并且分别持有WelfareRepository和View实例引用。
以start()初始化中调用的displayHome()方法为例,在具体的实现中,通过对两个实例的操作来进行相应的业务逻辑。简单来说,就是对Model层和View层实例提供的功能进行组合,使逻辑更加清晰,没有多余的代码。
(5)项目示例
通过仿照官方示例声明一些列的接口,实现了Model层-View层-Presenter层的分离,使代码和逻辑更加的清晰,具体源码托管在我的Github主页(点击查看)。