Dagger2+框架

716 阅读4分钟
链接: Dagger2+框架


今天要介绍给大家的是自己在使用dagger以及mvp框架时的一些思考, 希望提高我们在使用dagger以及mvp框架时的效率, 做到使用dagger时可以不用管dagger的那一套module,component, 使用mvp框架时, 对于presenter的实例不再维护而采用自动注入, 这样我们就可专心于业务代码.  而这,就是今天介绍的dagger2+框架.


背景介绍:

Dagger作为大项目中解耦的利器相信大家早有耳闻, 尤其是google接手后推出的dagger2, 更是用编译期生成代码替代原有的反射方案,不会给原本的应用造成性能上的影响。 Dagger这个利器优点很明显,但缺点也很明显:

1) 新手很难入手理解. 老司机就不说了,肯定一眼就明白了。 但新手门容易给绕进去了,每次注解产生的内部4个文件类很容易让新手蒙圈, 网上也一大堆关于dagger2入门,详解,原理分析的文章, 感兴趣的会耐着性子读下去(一遍又一遍哈), 直到理解(其实说到底就是间接的绕了一圈来做new对象这件事, 为啥要绕一圈呢,因为你用注解并不能修改源文件啊,所以只能间接的来咯), 耐不住性子的一般就放弃了.


2) 每次新对象(如activity或fragment)总需要在component和module上添加相应的代码,对象一多,component和module里就太多东西了,一不小心就漏了.


针对这些缺点, Dagger2+框架完全封装dagger, 让用dagger的用户察觉不到dagger的存在, 实现dagger完全解耦. 原理为自定义注解自动生成component, module 等dagger必须的文件,之后再通过dagger识别除产生出来的component和module文件再次产生dagger注入文件类.(注: 此框架结合mvp架构, 使Model, View层,Presenter层解耦, 使整个架构清晰少耦合)


1: 框架层简介:



2:

怎么做:

1) 需要java提供的注解框架(正如dagger的Module关键字一样), 我们提供AutoWire关键字, 来自动识别需要解耦的对象(如activity).


2) 在我们的AutoWireProcessor中, 自动检测出标注AutoWire注解的类, 对此类进行判断是activity或fragment,然后针对此类, 自动生成DIModule.java, DIComponent.java文件.



以上即为检测标注AutoWire的类, 判断是否为activity或fragment, 然后我们生成DIModule.java:



其中具体的生成provider的部分为:


同样的方式我们可以自动生成DIComponent文件. 最终生成文件:




可以看到与我们自己手写的并没区别,但它是自动生成的,自动维护的,也就是不会因为人为操作而失误的,节省了时间,仅这一点就已足够令人兴奋!

3) 经过前面两部,我们已经完全省去了每次新建对象都要在component和module里添加相应的方法, 终于可以给我们自己节省一杯咖啡的时间咯. 那么离最后的完全dagger隔离封装还有一步: 用dagger所必须的inject调用, 没有这步调用dagger也完成不了注入. 在这里我们巧妙的利用base的设计结合instanceof关键字来自动为大家的每个对象自动注入. (注: instanceof为编译期, 因此不会有任何的性能问题)

关键生成部分代码为:



那么会自动生成BaseAutoInjectActivity:




4) 最后,在我们的BaseCatActivity上,继承此BaseAutoInjectActivity, 同时BaseAutoInjectActivity继承BaseRxActivity,在BaseRxActivity的onCreate()中调用autoInject, 即可完成所有activity自动注入. 


4) 最后,在我们的BaseCatActivity上,继承此BaseAutoInjectActivity, 同时BaseAutoInjectActivity继承BaseRxActivity,在BaseRxActivity的onCreate()中调用autoInject, 即可完成所有activity自动注入. 






至此, 我们已经完成整个框架的初步设计, 将dagger完全隔离,同时进一步方便了MVP架构的开发. 详细的代码请参见github: Dagger2+框架. 欢迎star或fork. 



NEXT: 之后会在此框架上支持Service注解, 被注解的类自动支持单例, 省去每次getInstance判断为空的多余固定代码; 并且做到只要在一个类上标注@service, 即可直接使用. 无需遵循dagger2原有的在同一个类或其父类上再inject才可使用.


欢迎任何意见