仿Android官方MVP架构实现

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容