Hola,我是 yes。
这篇继续之前提到的 Dubbo SPI 来讲讲扩展点自适应机制。
这个名词听起来好像很高级,其实就是一个扩展代理类,通过参数返回对应的扩展实现类。
我写个代码看看应该就对扩展自适应一目了然了。
代码中的 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 来说比较有用,用来同时加载多个实现类,再看下官网的例子已经就比较清楚了。
原作者: 是Yes呀
原文链接:面试官问我什么是扩展自适应机制
原出处:yes的练级攻略
侵删