美文网首页
仿Android官方MVP架构实现

仿Android官方MVP架构实现

作者: quekx | 来源:发表于2016-05-06 18:38 被阅读693次

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

    相关文章

      网友评论

          本文标题:仿Android官方MVP架构实现

          本文链接:https://www.haomeiwen.com/subject/sgjjrttx.html