ARouter 源码浅析第二篇

服务管理

依赖注入

通过依赖注入的方式我们可以像上一篇文章中获取跳转参数一样获取服务,具体的实现我们还是通过源码来看一下吧。
当我们队一个自定义的service标注上@Autowired注解的时候,重新编译以后,APT会为我们自动生成如下代码:


image.png

至于inject方法在哪被调用上一篇文章已经分析过了,这里可以看到服务的注入跟参数的注入还是有区别的,具体的实现是插件帮我们做了,我节选了一点代码添上如下:


image.png

注意我圈出来的代码,这里区分了服务和普通类型分别生成不同的代码。而服务内部注入的实例还是通过普通方式获取的,也就是我下面要分析的第二种获取服务的方式。

普通方式

image.png

普通方式也分为2种,一种是根据name,另一种就是根据type。
下面只分析第一种根据name来获取服务的方式,因为根据type来获取实例是差不多的。

_ARouter#_navigation

private Object _navigation(final Context context, final Postcard postcard, final int requestCode, final NavigationCallback callback) {
        final Context currentContext = null == context ? mContext : context;

        switch (postcard.getType()) {
            case ACTIVITY:
                ……
                break;
            case PROVIDER:
                //注意这行代码
                return postcard.getProvider();
            case BOARDCAST:
            case CONTENT_PROVIDER:
            case FRAGMENT:
                Class fragmentMeta = postcard.getDestination();
                try {
                    Object instance = fragmentMeta.getConstructor().newInstance();
                    if (instance instanceof Fragment) {
                        ((Fragment) instance).setArguments(postcard.getExtras());
                    } else if (instance instanceof android.support.v4.app.Fragment) {
                        ((android.support.v4.app.Fragment) instance).setArguments(postcard.getExtras());
                    }

                    return instance;
                } catch (Exception ex) {
                    logger.error(Consts.TAG, "Fetch fragment instance error, " + TextUtils.formatStackTrace(ex.getStackTrace()));
                }
            case METHOD:
            case SERVICE:
            default:
                return null;
        }

        return null;
    }

最后会根据postcard.getType()类型来匹配PROVIDER,然后返回provider实例,而provider的set是在LogisticsCenter#completion中:

image.png

好了,实例获取到了其他的也就没什么好分析的了。
这里提醒一下,根据type来获取实例有局限性,只支持全局只有一个实现了相应接口的实现类,否则获取到的实例是系统扫描到的实现最后一个实例,同上一篇文章的PathReplaceService
why:那为什么根据type和根据那么来获取实例会有区别呢?通过源码我们发现type是从providersIndex Map中去获取实例的,而name是从providers Map中去获取实例的。相信到这大家应该已经明白了吧。
image.png

拦截器

我们在实现拦截器的时候需要实现IInterceptor,而IInterceptor也实现了IProvider,所以IInterceptor也是一个服务。所以interceptor的实例获取方式也是跟普通的service是一样的,这里不做分析。


image.png

对应上文的普通服务。
下面让我们来找一下具体得到拦截逻辑是在哪实现的呢?请看下面的代码:

InterceptorServiceImpl#_excute

InterceptorServiceImpl属于arouter核心服务类


image.png

InterceptorServiceImpl#doInterceptions

image.png

_ARouter#navigation

image.png

到这又回到我们上一篇文章页面跳转主流程的逻辑,他其实就是在真正跳转之前,先判断有没有符合的拦截器,有的话则先进行拦截器里的操作。这里要注意一点的是:拦截器是有优先级的,按照priority排序,priority越小优先级越高。这是因为
interceptorsIndex的类型是TreeMap,他具有以下两条特性:

1、TreeMap如不指定排序器,默认将按照key值进行升序排序,如果指定了排序器,则按照指定的排序器进行排序。
2、具体的排序规则,开发人员可以在int compare()方法中进行指定。

image.png

到此,拦截器的实现原理我们也就分析清楚了。

降级策略

我们知道使用系统自带的StartActivity()启动后就无法插手其中任何环节了,只能交给系统管理,这就导致了在跳转失败的情况下无法降级,而是会直接抛出运营级的异常,甚至导致崩溃,这个给用户的感觉就不是很好。所以ARouter就为我们提供了降级策略,主要分为2中方式,单独降级和全局降级(demo中这样称呼的)。

单独降级(接口回调)&全局降级

如下代码所示:


image.png

主要回调的就是onLost,那它又是在什么时候才被调用的呢?我们看如下代码


image.png

它在completion中主动抛出异常被捕获时调用的,如果callback不为null就会被调用,由此也可以看出单独降级的优先级是高于全局降级的,else中的代码就是全局降级的代码。使用方式也跟普通的服务是一样的这里就不分析了。
我们再看看LogisticsCenter.completion这个方法,这个是与编译期间生成的映射文件直接打交道的模块。先在加载到内存的节点仓库中查找是否有该目标节点的存在,没有就到组仓库中找,找不到就报错NoRouteFoundException,这就到了我们上面的逻辑中。

LogisticsCenter#completion

public synchronized static void completion(Postcard postcard) {
        if (null == postcard) {
            throw new NoRouteFoundException(TAG + "No postcard!");
        }

        RouteMeta routeMeta = Warehouse.routes.get(postcard.getPath());
        if (null == routeMeta) {    // Maybe its does't exist, or didn't load.
            Class<? extends IRouteGroup> groupMeta = Warehouse.groupsIndex.get(postcard.getGroup());  // Load route meta.
            if (null == groupMeta) {
                throw new NoRouteFoundException(TAG + "There is no route match the path [" + postcard.getPath() + "], in group [" + postcard.getGroup() + "]");
            }
        }
}

总结

经过两篇文章,我们基本上已经把ARouter的主要功能都分析了。其实看着很神奇的功能,只要我们深入到源码里就会发现其实也就那么回事。不过我们还是能从优秀得到框架中学到很多优秀的设计思想和前沿的技术的,比如IOC、APT, AOP等

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容