Dagger2+框架

链接: 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:   框架层简介:

Dagger2+框架_第1张图片


2:  

怎么做:

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

@Target({ElementType.TYPE})

@Retention(RetentionPolicy.CLASS)

public @interface AutoWire {

Class presenter();

Class contract() default Object.class;

}



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


Set elements=roundEnvironment.getElementsAnnotatedWith(AutoWire.class);

Set activityElements=new HashSet<>();

Set fragmentElements=new HashSet<>();

for(Element element : elements) {

if(isSubtypeOfActivity(element.asType(), mMessager)) {

activityElements.add(element);

}elseif(isSubtypeOfFragment(element.asType(), mMessager)) {

fragmentElements.add(element);

}

}

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

Dagger2+框架_第2张图片

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


Dagger2+框架_第3张图片

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


Dagger2+框架_第4张图片
Dagger2+框架_第5张图片

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

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

关键生成部分代码为:

Dagger2+框架_第6张图片

那么会自动生成BaseAutoInjectActivity:


Dagger2+框架_第7张图片


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


Dagger2+框架_第8张图片

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

你可能感兴趣的:(Dagger2+框架)