写点什么

面试官问我什么是扩展自适应机制

  • 2021 年 11 月 12 日
  • 本文字数:1581 字

    阅读完需:约 5 分钟

注解在方法上表明 Dubbo 需要为这个方法生成代理逻辑。


拿上面提到的 AdaptiveYes 类来说,如果这个类上被标注了 @Adaptive 那么说明这个类就是 Yes 这个扩展要用的代理类,框架就不用动态生成了。



如果 @Adaptive 被标记在接口 Yes 的 sayHi 这个方法上,那 Dubbo 就需要用字节码工具来生成 AdaptiveYes 这个代理类。



在 Dubbo 中,类上被修饰 @Adaptive 只有两个,分别是 AdaptiveCompiler(自适应选择编译器实现)和 AdaptiveExtensionFactory(自适应选择扩展工厂)


还记得之前提到的 Dubbo 自动注入功能的代码嘛?就是通过 SPI 找到的扩展实现类内部需要注入对象的功能。



当时留了个坑,现在填上。


这行代码是要通过扩展实现类 set 方法上的参数找到扩展点要注入的对象,而这个 objectFactory 就是自适应扩展代理类。



Dubbo 中的注入相对 Spring 而言比较复杂,因为有可能需要注入的是 Dubbo 中其它自适应扩展对象,也有可能注入的是 Spring Bean,或者是我们自行定义的容器里面的对象等等。


所以依赖注入的对象需要去多处查找,因此加了一层,搞了个自适应代理扩展类。


在 Dubbo 中的 ExtensionFactory (扩展工厂,从工厂中查找要注入的对象)有三个实现:



  • SpringExtensionFactory:从 Spring 容器中去加载 Extension

  • SpiExtensionFactory:Dubbo 自己的 SPI 去加载 Extension

  • AdaptiveExtensionFactory: 自适应的 AdaptiveExtensionLoader,也就是我们上面提到的代理类,由人工编写的。


ExtensionLoader 中的 objectFactory 用的就是 AdaptiveExtensionFactory 这个实现类了,咱们跑起来打个断点看看。



嗯,确实是,还能看到 AdaptiveExtensionFactory 的成员变量 factories 还保存了另外两个工厂。


我们来简单地看下 AdaptiveExtensionFactory 。


这个工厂会先去加载所有 ExtensionFactory 的扩展类,然后查找 Extension 的时候会遍历每个 ExtensionFactory 实现类去找要注入的对象,找到了就返回。



所以 Dubbo 就是通过这种方式来实现 IOC 的注入,很粗暴简单,每个工厂遍历过去查找需要注入的对象。


好了,填了之前文章 Dubbo IOC 的坑,也讲了下 @Adaptive 修饰类的情况(就是直接把这个类作为代理类)。


接下来要讲讲修饰方法的情况,相对而言比修饰类要复杂。


不过也不难,无非就是多了几步,要用字节码工具生成代理类的源码,然后编译成 Java 字节码,然后加载到 JVM 中,就是这样。


我们来看看源码,入口就是 getAdaptiveExtension 方法。



那个 cachedApaptiveClass 就是 SPI 扫描对应文件夹加载类的时候记录的。



结合上面两个代码图就知晓为什么类上标注 @Adaptive 的时候直接就用那个类,不然就需要框架生成代理类了。


我们再来看看框架生成的代码是怎样的。



我们看的是 Protocol (协议接口,Dubbo 支持很多协议,默认 dubbo 协议)的自适应扩展代码,我们先看下 Protocol 这个接口的定义,然后再看看生成的代码。




如何


【一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义】
浏览器打开:qq.cn.hn/FTf 免费领取
复制代码


生成上面 code 内容的方法我就不分析了,反正就是各种判断然后字符串拼接而成的,至于编译之前也提到了,Dubbo 默认选的是 javassist。


至此整个自适应逻辑扩展已经很清晰了,然后上完整 SPI 的图,相信看了图之后整个流程就了然于心了!



Dubbo 中的 Activate




再提一提 @Activate ,这个就不进行源码分析了,此注解是用来实现自动激活特性的。



主要的参数是:


  • group:表明类得在 Provider 端被激活还是在 Consumer 端被激活。

  • value:URL 参数上出现指定的值被激活。

  • order:扩展激活类之间的排序。


简单地说就是标注了这个注解的扩展会被记录,然后调用的时候根据参数来选取合适的扩展实现类。


比如参数的 group 和当前扩展类的 group 匹配,出现了指定的 key ,然后就会被激活。


对于 Filter 或者一些 Listener 来说比较有用,用来同时加载多个实现类,再看下官网的例子已经就比较清楚了。

评论

发布
暂无评论
面试官问我什么是扩展自适应机制