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

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

我写个代码看看应该就对扩展自适应一目了然了。
在这里插入图片描述
代码中的 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 来说比较有用,用来同时加载多个实现类,再看下官网的例子已经就比较清楚了。

在这里插入图片描述

读者福利

感谢你看到了这里!
我这边整理很多2021最新Java面试题(含答案)和Java学习笔记,如下图
在这里插入图片描述

上述的面试题答案小编都整理成文档笔记。 同时也还整理了一些面试资料&最新2021收集的一些大厂的面试真题(都整理成文档,小部分截图)免费分享给大家,有需要的可以 点击进入暗号:CSDN!免费分享~

如果喜欢本篇文章,欢迎转发、点赞。

记得关注我!

猜你喜欢

转载自blog.csdn.net/weixin_49527334/article/details/113893902