Android面试经验Android开发经验谈android技术专栏

Android适合组件化开发的路由框架:Launch

2017-07-20  本文已影响436人  五香鱼cc

1.概述

最近越来越不想写代码了,特别是一些重复性的代码,比如由于每次启动一个 Activity,我们都会很习惯的在 Activity 中写下:

public static void launch(Activity activity) {
    Intent intent = new Intent();
    intent.setClass(activity, xxxActivity.class);
    activity.startActivity();
}

已经有两年Android开发经验的我掐指一算,好像有很多方法可以直接帮咱们生成这样类似的代码:

本文的重点放在了编译时注解的框架实现,毕竟是要和时代接轨的啦!

2.需求剖析

我的需求很简单,就完成通过注解来代替Intent代码来启动 Activity,让注解来帮我们生成 launch 方法,而且还要能够在组件化开发中使用。
需求很明确,所以直接开始构思:
所有的 launch 方法其实很类似,不同的就是跳转的终点 xxxActivity ,那么我们就想办法来构造一个对应的关系:通过配置注解 LaunchAnn 来给每个 Activity 分配一个String别名,我通过找到这个别名,我就知道了这个对应的真实 Activity。这样在需要启动xxxActivity的地方,我通过调用别名来启动这个 xxxActivity 就可以了,这样启动就不会依赖 xxxActivity,而仅仅是一个别名字符串而已。

3.开发流程

创建三个 module:

开发注解框架可以参考:Android 如何编写基于编译时注解的项目。注意的点:

定义注解:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.CLASS)
public @interface LaunchAnn {
    String value();
}

注解使用:

@LaunchAnn("RuntimeActivity")
public class RuntimeActivity extends Activity {
    
}

打开 activity:

public void onClick(View view) {
    Launch.launch("RuntimeActivity", this);
}

iocCompiler 中的 CakeProcessor 中(AbstractProcessor实现类),生成了一个固定的LaunchUtil工具类。通过遍历有哪些类使用了LaunchAnn注解,然后把 LaunchAnn 的 value 值(这里的“RuntimeActivity”)作为 key,类对应的全路径(包名+类名)作为 value 存入到 LaunchUtil 工具类的 map 中,这样我就可以直接通过注解 LaunchAnn 的 value 值可以获取到对应类的全路径。
iocApi 的作用就是提供给用户调用的接口(onClick中的内容调用),LaunchUtil 中包含了注解对应的map信息,所以关键点就是能够解析LaunchUtil中的内容。
由于 iocApi 和 iocCompiler 没有半毛钱联系,所以怎么能够获取到 LaunchUtil 这个类?这点还是让我花费了一些时间的,最后通过在 iocApi 中定义一个 LaunchInjector 接口,实现通过key值获取Class的getPackageName的接口。当然我们的 LaunchUtil 工具也要实现与LaunchInject这个接口,这样通过反射LaunchUtil的,直接形成了一个 LaunchInjector 的对象,通过 getPackageName 方法就可以获取到需要跳转的 Activity 对应的 Class。
我感觉介绍的差不多了,不懂就自己看代码了。

4.适配组件化方案

其实最开始仅仅是想做一个懒人开发者,这样我就不用每次都写 launch 了,但是开发到最后才发现,前面还有好多要去学习的地方。
由于本身自己在学组件化开发,所以想把写的注解运用到组件化当中去。有4个 module:modulebase、moduleA、moduleB、app。命名中可以看出 modulebase 是基础包,moduleA 和 moduleB 都是功能模块包,最后总包 app module。
由于每个模块都是用了Launch,所以我们可以发现在编译注解的时候,每个 lib 下面都会有一个 LaunchUtil,在合并的时候就会冲突(提示类重复),所以我们需要针对不同的模块,定制不同的类名,我这边是根据模块绑定了,例如:LaunchUtil$$moduleA.java。这样多个 LauncUtil 类就可以共存了。
现在由于有多个 LauncUtil,我们怎么把所有的 LaunchUtil 里面的map信息进行汇总,得到一个总的 map 信息呢?这个东西我想了很久很久,过程中经历了两种实现,其中第一种最后证明不行,最后选择了第二种。为啥还要说第一种呢,因为我感觉还是有必要记录一下的:

5.总结

前面的编译注解对老司机来说其实都是比较简单的,开发流程表比较固定。对于适配组件化开发的过程,其实还是一个比较新鲜的事情。因为看到了阿里的ARouter是适合组件化开发的,所以自己也是有点蠢蠢欲动的赶脚。最后有一个可以让人参考的 demo,希望大家多多交流和学习。对于还有更好的方法来获取所有的 LaunchUtil 的,欢迎交流。
存在的不足:

上一篇 下一篇

猜你喜欢

热点阅读