链接: 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).
@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<? extends element> elements=roundEnvironment.getElementsAnnotatedWith(AutoWire.class);
Set<Element> activityElements=new HashSet<>();
Set<Element> 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:
其中具体的生成provider的部分为:
同样的方式我们可以自动生成DIComponent文件. 最终生成文件:
可以看到与我们自己手写的并没区别,但它是自动生成的,自动维护的,也就是不会因为人为操作而失误的,节省了时间,仅这一点就已足够令人兴奋!
3) 经过前面两部,我们已经完全省去了每次新建对象都要在component和module里添加相应的方法, 终于可以给我们自己节省一杯咖啡的时间咯. 那么离最后的完全dagger隔离封装还有一步: 用dagger所必须的inject调用, 没有这步调用dagger也完成不了注入. 在这里我们巧妙的利用base的设计结合instanceof关键字来自动为大家的每个对象自动注入. (注: instanceof为编译期, 因此不会有任何的性能问题)
关键生成部分代码为:
那么会自动生成BaseAutoInjectActivity:
4) 最后,在我们的BaseCatActivity上,继承此BaseAutoInjectActivity, 同时BaseAutoInjectActivity继承BaseRxActivity,在BaseRxActivity的onCreate()中调用autoInject, 即可完成所有activity自动注入.
至此, 我们已经完成整个框架的初步设计, 将dagger完全隔离,同时进一步方便了MVP架构的开发. 详细的代码请参见github: Dagger2+框架. 欢迎star或fork.