面试官问我什么是扩展自适应机制
这个名词听起来好像很高级,其实就是一个扩展代理类,通过参数返回对应的扩展实现类。
我写个代码看看应该就对扩展自适应一目了然了。
代码中的 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 来说比较有用,用来同时加载多个实现类,再看下官网的例子已经就比较清楚了。
往期推荐
点击二维码识别关注
点在看,让更多看见。