查看原文
其他

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

IT服务圈儿 2022-09-10

The following article is from yes的练级攻略 Author 是Yes呀


作者丨是Yes呀

来源丨经授权转自公众号yes的练级攻略(ID:yes_java)


这个名词听起来好像很高级,其实就是一个扩展代理类,通过参数返回对应的扩展实现类。

我写个代码看看应该就对扩展自适应一目了然了。

代码中的 AdaptiveYes 就是代理类,实现同样的接口,然后根据调用时候的参数去选取对应的实现类进行调用,这就是扩展自适应。

例如拿到的yesName 是“yesA”则返回YesA这个实现类,是“yesB”则返回YesB这个实现类

是不是没什么花头?就简单加了一层,可以根据请求的参数来动态选择对应的扩展实现类,让扩展更加灵活

理解了什么是扩展自适应之后,我们再来具体看看 Dubbo 中的实现。

Dubbo 中的 Adaptive 注解

从代码中可以看到 Adaptive 可以注解到类上或方法上。

注解到类上的话表明这个类就是要用的代理类,所以 Dubbo 不需要用字节码工具为这个扩展生成代理类。

注解在方法上表明 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 这个接口的定义,然后再看看生成的代码。

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

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

Dubbo 中的 Activate

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

主要的参数是:

  • group:表明类得在 Provider 端被激活还是在 Consumer 端被激活。
  • value:URL 参数上出现指定的值被激活。
  • order:扩展激活类之间的排序。

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

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

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

有偿征稿IT服务圈儿正式开启投稿通道,稿费:60~5000元不等,长期有效!!!
点此查看详情


1、Vuex 4 正式发布

2、这几个电脑的误区,你信过几个?

3、五个吊炸天的网站

4、JetBrains 2020年度亮点:IDEA中国用户最多、持续加大开源贡献力度

识别关注我们

了解更多精彩内容

点分享

点点赞

点在看

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存