仿Android官方MVP架构实现

2016-05-06  本文已影响693人  quekx

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主页(点击查看)
上一篇下一篇

猜你喜欢

热点阅读