传统MVP用在项目中是真的方便还是累赘?

前言(最后奉上福利)

自从Google在去年放出MVP官方Sample后,越来越多的人开始加入MVP大军,MVP可谓在16年大放异彩,我也乘势推出了我的MVP框架狂刷了一波存在感

问题

但在使用当中我也发现了诸多弊端,导致很多初学者,在写过Sample后,就再也没在自己的项目中使用过MVP

MVP需要创建太多的类和接口,并且每次通信都需要繁琐的通过接口传递信息

这是大多数使用过MVP的朋友,最能感受到的,最近在帮公司技术面,我也时常问应聘者,能否尝试着解决这些问题?

解决方案

其实我之前已经有一套解决方案,其实也不能叫解决,只能说是缓解😅

硬解决

所谓硬解决,便是使用比较暴力的方式😂,通过Template自动生成需要的类和接口,这样少去了频繁的复制粘贴

软解决

所谓软解决,那就要动动脑子,稍微优雅的解决了😀

  1. 对于逻辑简单的页面可以不使用Presenter,直接在ActivityFragment中处理逻辑,在Presenter中如果不需要处理数据,也可以不实用Model

  2. PresenterModel都可以无限制的重用,所以MVP的划分不需要太细粒度,稍微粗粒度一点,即不需要每个ActivityFragment都给他划分一套MVP,可以几个ActivityFragment使用同一个Presenter(使用同一个类不是同一个对象,这个Presenter含有可以共用的逻辑),也可一个ActivityFragment根据不同的需求持有多个不同类型的Presenter对象,Model层同理,这样灵活使用,可以在一定程度上缓解MVP类和接口较多的缺点

并没有完全解决问题

通过上面的解决方案,是可以一定的缓解MVP的缺点,但是并不能完全解决上述缺点

比如想重用Presenter,Presenter就必须只含有公用的逻辑,而实际项目中公用的逻辑并不是那么多,所以能减少的类和接口也是很有限的,如果强制将不同页面的逻辑放在同一个Prsenter中,来达到重用的目的,那么每个Activity会被迫实现许多并不需要的方法,得不偿失

寻求解决方法

因此我看了大多数MVP框架,寻求如何彻底改善这个问题,像支付宝团队使用的TheMVP框架,是通过将ActivityFragment作为Presenter,将UI操作抽到Delegate中,作为View

TheMVP优点

这样做的好处是,不仅可以少写很多类,而且Presenter直接就可以和ActivityFragment的生命周期做绑定,且可以随便重用View(但大多数场景都是重用Presenter,因为View层变化总是比其它层频繁)

TheMVP缺点

缺点就是不能重用Presenter,并且对于Presenter的实现有限制,必须是ActivityFragment,如果要在其他地方实现Presenter,如Adapter,Dialog就必须根据它的特性重新写对应的Presenter基类

因为Presenter基类继承了ActivityFragment,如果我们需要通过继承使用其他ActivityFragment,那就又需要修改Presenter基类,一旦某个Activity需要继承其他不同的Activity,那又需要重新创建一个基于此ActivityPresenter基类,导致一个ActivityFragment有多个不同的Presenter基类

分析问题,解决问题

总结一下MVP的缺点

1.粒度不好控制,控制不好就需要写过多的类和接口
2.如要重用presenter可能会实现过多不需要的接口
3.Presenter和View通过接口通信太繁琐,一旦View层需要的数据变化,那么对应的接口就需要更改

想要在根本上解决以上问题,我想必须换个思路,能不能通过改变传统MVP架构来解决这些问题?

实现MVP现阶段有两种方式,各有优缺点:

一个是将ActivityFragment作为Presenter,抽象一个View层出来

一个是将ActivityFragment作为View,抽象一个Presenter层出来

我想达到重用Presenter的目的,自然选择了后者

在某一天我突然想到了Handler,他只通过一个handleMessage方法,根据Messagewhat字段处理不同的操作,这样向上层提供一个统一的入口,下层不管如何改变并不会影响上层,并且同样可以实现多种的操作

于是根据这个思想,我重新改造了MVP架构,让Presenter通过MessageView层通信

如何实现

先上张图

具体做法是,VIEW层持有Presenter对象,当用户请求一个事件,则调用Presenter中的方法,并把持有View引用Message传给此方法,此方法处理完请求逻辑后将数据封装到Message中,并通过Message持有的View引用回调ViewhandleMessage方法,让View做不同的操作,最后释放掉Message的所有引用,放入消息池

Presenter并不直接持有View,方法执行完即表示和View的关系解除

Handler的原理很像,Handler是将消息放入MessageQueue,Looper去轮循处理消息,我这里是将消息放入,Presenter的方法,并立即处理消息

总结

这样就能解决上述的缺点:

  1. 少写了很多类和接口

  2. 并且Presenter只需要通过handleMessage一个方法与View通信,也就不用繁琐的一直添加接口方法,只需要一个Message参数,通过Message封装数据,即使View需要的数据类型发生改变,也不需要更改任何方法,所以也不会影响上层调用

  3. 随便重用Presenter,即使你一个Activity,重用10个不同的Presenter,那也只用实现一个handleMessage方法,不需要实现View中其他用不到的方法,通过一个方法同样能做到不同的操作(传统MVP一个页面对应一个Presenter,其实大多数Presenter只有一两个方法,这样导致存在大量代码寥寥无几的Presenter,你有想过将相近的逻辑都写到一个Presenter中,一直重用Presenter有多爽吗😁)

  4. Presenter中的方法需要Activity传递一些数据时,也可以将数据封装到Message中传给Presenter,这样即使需要的数据类型发生改变,也不需要更改方法,所以也不会影响上层调用

只有能不断的灵活重用,才能感受到MVP的强大之处

当然很多不同的逻辑都写在一个Presenter中,虽然可以少写很多类,但是后面的扩展性肯定不好,所以这个粒度需要自己控制,但是对于外包项目简直是福音

说了这么多还是要看看Demo,具体该怎么做吧?

Go!觉得好一定要右上角Star哦!

Tips

现在的框架是一个比较轻量级的框架,没有网络层,后面会开一个分支提交比较完整的框架和文档,像MVPArms一样

– The end

Word 2016

2016年,对我来说是最重要的一年,也是我价值体现最大的一年

为什么这么说呢?

生活上我本是一个三点一线,即无情趣的人(也许这是程序员的本质吧),时刻保持着学习,技术进步是我最大的乐趣,偶尔看看综艺节目,偶尔和朋友5黑(注孤生)

因为上一家公司的关系(做健身相关的产品),我接触到了健身,并在年初办了健身卡

没错和大部分办健身卡的少年一样,只保持了一个星期的热情,虽然后面没有放弃,但是每次运动量都不是很多,效果自然不理想

我期待自己的身体越来越好

我期待自己的身材越来越棒

我期待自己的毅力越来越强

但是又有什么用呢,梦想谁都有,道理谁都懂,又有几个坚持做下去,并成功的呢?

作为一个以后想干大事的人,我告诉自己人与人的差别就是在于做事的态度以及考虑事情的角度

有些人他渴望这个梦想,想了想需要经历的苦难,我的天这么难,猜测自己就算坚持了,后面也会放弃吧?我还是别白白浪费时间了

有些人他渴望这个梦想,想了想需要经历的苦难,哈哈,有点挑战,没准我成了呢?于是开始坚持了一段时间,呵?以我的条件后面可能还是不具备竞争力吧终将被淘汰,于是没有了然后

有些人他渴望这个梦想,想了想需要经历的苦难,他自信的告诉自己一定能成功,制定计划,踏出第一步,坚持下去,时刻努力,并告诉自己能行的

于是他成功了?

获得了利益

于是他失败了?

获得了经验

不亏啊?

来干了这碗鸡汤!

没错我就是第三种,我成功的坚持了下来,用汗水换来了8块腹肌和人鱼线(对于健身这个只要花时间和汗水,就一定会有成就的事情都办不成,还办大事?)

工作上,在年初,有幸进入一家互联网公司

这并没有什么值得奇怪的啊?

在这家公司,我第一次接触了直播领域,在之前我的心目中,直播是高大上的领域,没有一点技术沉淀是不敢触碰的,直播涉及到的高并发,延迟,弱网环境的优化,不是一般的公司能解决的,还好在使用了七牛的解决方案后,让这个难题迎刃而解

同时我也第一次意识到框架对一个项目的重要性,之前的项目从来都是一个Activity写到底,有什么变更直接在里面做操作

也许只有有经验的开发者,会在项目不断的增大,以及项目需求的频繁更改,从而导致代码越来越难看,越来越难以维护以及扩展,而领悟框架的重要性

庆幸我及早的领悟了

于是我开始收集各种资料,在我的屏幕上出现了:

MVP

Dagger2

RxJava

Retrofit

EventBus

现在又有一个难题,这些技术我都是第一次接触,如何让他们融合起来,适应自己项目的需求,于是又是各种实验,就有了现在MVPArms的雏形

在某天突发奇想,这些技术门槛高,集成也困难,我能不能让后面的所有项目,都非常轻松的引入这个框架?

受开源社区的影响,我不仅写出来了,还将它开源,从而让更多的小伙伴受益,很庆幸在我和小伙伴们的努力下,截止目前star数已经超过1k

开源项目也让我从关注如何用别人的代码,到关注别人怎么通过代码实现

再次感谢之前的直播项目,让我了解到了sdk,以及sdk的编写和文档的编写,

让我从只关注当前需求,到关注代码后续的扩展

让我从只关注代码的功能实现,到关注代码的封装性

让我从只关注自己的代码自己是否好用,到关注别人用起是否舒适易懂

让我从只关注一个产品的最终交付,到关注产品文档编写以及后期维护

虽然这个产品上线后没能达到预期的效果,但是我非常感谢这个产品教会了我这么多外面很难接触到的知识

直播

im

框架构建

文档编写

SDK集成

我庆幸自己在不大的年纪,通过自学换来了不错的收入

我庆幸自己在不大的年纪,通过汗水换来了不错的身材

我庆幸自己在不大的年纪,通过实践知道了自己的方向

没错2017年我还是要干大事情,我还是要不断改变自己,让自己成熟,有担当,细致,我知道自己脸皮薄,所以我希望2017年我能上台演讲!(不开玩笑😂)

别人说改变自己就是给自己做对,就是让自己的生存环境不断的从舒适变为糟糕,就是不断的拿刀子虐待自己

没错我就是喜欢自虐

感谢自己的努力与坚持

感谢曾经提供给我帮助的小伙伴和公司

2017继续努力,勿忘初心

– The end