我项目中的架构——MVP

排版不佳建议查看原文

浅谈MVC


        在我待过的小公司中,大多数Android项目依然使用的还是MVC架构,因为它相对简单,项目搭建初期,我们可以不用花费太多精力便可以搭建出项目框架,快速投入业务代码的开发中。随着时间的推移,业务的增加,代码层次越来越不清晰,甚至会出现一个Activity超过两千行的情况,包含了操作网络的回调函数、解析数据的函数、本地存储的函数、以及操作UI的函数。随之而来更难调试的bug、内存泄漏等等问题。相信你一定看见过。在我看来这种项目连MVC架构都不是!我们就叫它VC架构好了。

我项目中的MVP


        由于曾经就职某小外包公司的缘故,平均每隔1个月我就会从0开启一个新项目,所以我开始尝试MVP、甚至是MVVM架构。

        总体来说,我的目的是使得项目层次更清楚,在View层抽离一切于UI操作无关的任务,移植到Model层独立,并且中间新增Presenter关联View与Model层。

View层



        第一步是建立View接口,目的是定义我们的页面有哪些公共的UI行为,可见图上showLoading函数与hideLoading函数定义为我们页面加载数据时操作进度条的函数。showError函数为加载数据失败情况UI的操作函数。onSuccess为加载数据成功的操作函数,UI可直接获取形参中的数据包显示操作。isActive代表当前页面是否可见。以上统称为View接口,由View(Activity、fragment)实现。

        第二步建立Contract接口,你可能会问Contract是什么,由于我们定义了BaseView接口抽离了View的公共UI行为,可是每个页面都可能有自己的UI行为,所以BaseView接口只能是基类,由每个页面的View接口去继承它,如图下的LoginView接口。那么一个页面将要新建一个View接口文件,会非常多,所以我们将所有View接口写到了Contract接口内部,Contract可以理解为一个装载接口的容器。


        第三步,创建View接口的实现类LoginActivity!


        图上,我们实现了LoginView接口的所有函数,如showLoading我们只需要显示progressDialog,无需关心何时调用,只关心函数需要实现的UI操作即可,职责单一。

Model层


        有了View层,下面实现Model层。


        第一步,历史总是惊人相似,图上定义LoadTasksCallBack接口与BaseView接口十分相似,目的是定义我们数据请求的4种基本状态,onSuccess加载成功、onStart开始加载、onFailed加载失败、onFinish加载结束。


        第二步,图上定义BaseNetTask接口,目的是让负责逻辑代码的Model层拥有执行任务的函数execute。我们发现其形参中携带了我们上面定义的LoadTasksCallBack,目的是让execute函数开始执行或完成后通过其形参LoadTasksCallBack接口回调其执行结果给LoadTasksCallBack接口的实现类。从而实现与Presenter层通信。


        第三步,与View层的BaseView类似,我们的BaseNetTask仅仅只是定义了一个基本的执行任务的函数execute,也许一个页面并不只有一次或者一种逻辑任务需要处理,这时候execute显得不够了,所以BaseNetTask仅仅作为基类,如图上的LoginNetTask在BaseNetTask基础上扩展了自己的任务函数getTime。最后同样定义在Contract内部。


        第四步,创建实现LoginNetTask接口的实现类LoginTask。如图上红色框框标注位置,我们实现了execute函数与getTime函数,同样是职责单一,不关心谁会调用,仅仅关注内部实现,execute中我们完成了登陆的逻辑,并且通过LoadTasksCallBack接口成功将结果回调。这样实现LoadTasksCallBack接口的实现类则会收到此次任务的结果。

        还有值得注意的是,Task相关的类都采用单例模式。

Presenter层


        经历了View层与Model层,你会发现它们都是相对独立的模块,各自并不直接联系。所以我们需要创建BasePresenter如图下:


        第一步,我们实现了LoadTasksCallBack接口,在Model层我们知道LoadTasksCallBack接口主要让Model层execute函数开始执行或完成后通过其形参LoadTasksCallBack接口回调其执行结果给LoadTasksCallBack接口的实现类。没错,就是这里,Model层的任务执行结果都来到这。

        在构造函数中我们发现,BasePresenter需要接收BaseView接口的实现类与BaseNetTask接口实现类才可被实例化。换句话说,创建BasePresenter需要传Activity||fragment与LoginTask。

        由于LoginTask所有任务的结果都会传递到BasePresenter,所以在onSuccess中,我们判断页面是否可见,然后隐藏进度条,再转发数据到View层的onSucess中。目的就是Presenter收到Model层的结果回调后,再根据响应状态回调View层UI操作函数。其余函数职责相同。


        第二步,图上你可能会问,咋又是BasePresenter,它们还真不一样,定义BasePresenter接口,目的是提供View层去调用加载数据函数的一个入口,如Activity通过startExecute方法开启一个加载数据的任务,具体往下看:


        一样的目的,因为BasePresenter接口只定义一个提供给View层操作的基本加载数据的函数入口,所以只能定义为基类,每个页面可以自定义增加提供给View触发,加载数据的函数。如上图LoginPresenter,由于LoginNetTask提供getTime函数,如果View层需要触发getTime,则需要在LoginPresenter相应增加getTime函数。


        第四步,图上创建LoginPresenter类继承BasePresenter,实现LoginPresenter接口。是不是信息量有点大?首先继承BasePresenter类,这样Model层的结果回掉统统都在基类中处理完毕,派生类只关心View层的任务。

       然后实现LoginPresenter接口,startExecute函数与getTime函数职责就是接受View层的调用,其内部再去转发调用Model层的execute与geiTime函数触发数据操作,起到了中介的作用。

        最后:在View层的Activity初始化时候,我们仅仅只需要实例化Presenter,通过Presenter我们可以调用任意Model层的函数,并且最终Activity-onSuccess中只获取加载处理完毕的数据直接更新UI,并不关心其内部实现。


总结



问题


        onSccess函数每个页面只有一个,只能实现一次。由于onSccess函数承担了所有的数据请求结果的接收,假设一个页面有多种多个不同的数据请求结果,则在onSccess(Object data),data中无法区分数据造成严重问题。小编也是非常苦恼。后来将函数形参修改成onSccess(Object data,int Type),新增了Type,在Model层加以Type区分,则在View层根据Type分别更新UI。

这可能不是最佳解决方案,如果你知道可以给我留言,谢谢支持。

欢迎长按下图-识别图中二维码或者扫一扫,搜索微信公众号:黄君华。关注我的公众号:


如果你有不同意见或建议或者有好的技术文章想和大家分享欢迎投稿,可以把你的文章使用附件的形式发送到我的邮箱2908116133@qq.com 谢谢阅读!

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

推荐阅读更多精彩内容