SPI
SPI只是一种协议,它只是规定在META-INF目录下提供接口的实现描述文件,由框架本身定义接口、规范,第三方只需要将自己实现在META-INF下描述清楚,那么框架就会自动加载你的实现。比如Dubbo的规则是在META-INF/dubbo、META-INF/dubbo/internal或者META-INF/services下面以需要实现的接口全面去创建一个文件,并且在文件中以properties规则一样配置实现类的全面以及分配实现一个名称。
ExtensionLoader
Dubbo对于SPI的实现全部集中在类ExtensionLoader中,ExtensionLoader是一个单例工厂类,它对外暴露getExtensionLoader静态方法返回一个ExtensionLoader实体,这个方法的入参是一个Class类型,这个方法的意思是返回某个接口的ExtensionLoader,对于一个接口,只会有一个ExtensionLoader实体。
ExtensionLoader实体对外暴露一些接口来获取扩展实现,这些接口分为几类,分别是activate extension, adaptive extension, default extension, get extension by name , supported extension。
(1)URL为总线的模式
即运行过程中所有的状态信息都可以通过URL来获取。
(2)activate extension
activate extension都需要传入url参数,这里涉及到Activate 注解,这个注解主要用于标注在插件接口实现类上,用来配置该扩展实现类激活条件。
在Dubbo框架里 面的Filter的各种实现类都通过Activate标注,用来描述该Filter什么时候生效。
另个activate注解还有一个参数order,这是表示一种排序规则,因为一个接口的实现有多种,返回的结果是一个列表,如果不指定排序规则,那么可能列表的排序不可控,其中order的值越大,那么该扩展实现排序就越靠前,对于排序还可以使用before和after来配置。对于用户自定义的扩展默认是追加到列表后面的。
getActivateExtension(URL url, String[] values, String group)
(3)adaptive extension
Dubbo框架提供的各种接口均有很多种类的实现,为了能够适配一个接口的各种实现,便有了adaptive extension。createAdaptiveExtensionClassCode。
对于这种途径,Dubbo也提供了一个注解Adaptive,用来标注在接口的某个实现上,表示这个实现并不是提供具体业务支持,而是作为该接口的适配器。
Dubbo框架中的AdaptiveExtensionFatory就是使用Adaptive进行了标注,它用来适ExtensionFactory接口SPIExtensionFactory和SpringExtensionFactory两种实现 ,它会根据支行的状态来确定具体调用ExtensionFactory的哪个实现。
ExtensionLoader通过分配接口配置的adaptive规则动态生成adaptive类并且加载到ClassLoader中,来实现动态适配。adaptive注解有一个value属性,通过设置这个属性便可以设置该接口的Adaptive的规则,Dubbo动态生成Adaptive的扩展接口的方法入参必须包含URL或者参数存在能返回URL对象的方法,这样才能根据支行状态动态选择具体实现。
(4)get extension by name
就是通过接口实现的别名来获取某个具体的服务。
(5)default extension
被实现的接口必须标注SPI注解,用来告诉Dubbo这个接口是通过SPI来进行扩展实现的,在接口上标注SPI注解的时候可以配置一个value属性用来描述这个接口的默认实现别名。
应用场景
在dubbo中随处可见ExtensionLoader的使用,几乎任何预留扩展的服务都通过ExtensionLoader加载。
以ReferencConfig为例,其中的Protocol等便是通过ExtensionLoader来加载的。
(1)首先通过getExtensionLoader方法获取对应的Protocol的ExtensionLoader实例。
(2)之后便是通过getAdaptiveExtension()获取对应的Adaptive实现。
(3)Adaptive实例不存在,则通过createAdaptiveExtension()创建,在这之前要加载Adaptive Class。
(4)如果找不到被Adaptive注解的适配器,则通过dubbo动态创建(字节码)适配器。
(5)由injectExtension完成依赖服务注入(借助ExtensionFactory找到依赖的服务,通过set方法注入)。
Dubbo的扩展点主要有adaptive和wrapper两种:
(1)adaptive,因为dubbo底层会大量使用反射,出于性能考虑默认使用javassist字节码编译生成一个adaptive,由它动态委派处理。用户可以自己实现一个adaptive,只需要对某个类打上@adaptive即可。对于默认编译生成Adaptive的方案,需要使用@Adaptive声明接口上的哪些方法是adaptive方法。扩展点名称的key默认是接口类型上@SPI#value,方法上的@Adaptive#value有更高优先级。
(2)包装类必须有一个参数为spi接口类型的构造函数,否则不能正常工作。判断warpper的标准是class有没有一个参数为接口类型的构造参数。Wrapper可以有多个,会被按顺序依次覆盖,假设spi定义如下:
A=a.b.c
B=a.b.wrapper1
C=a.b.wrapper2
wrapper的最终结构则为B-C-A
最后欢迎大家访问我的个人网站:1024s