spi(service provider interface)是jdk内置的一种服务提供发现机制,可以用来启用框架扩展和替换组件,主要用于框架中开发,例如dubbo、spring、common-logging,jdbc等采用采用spi机制,针对同一接口采用不同的实现提供给不同的用户,从而提高了框架的扩展性。
java spi实现
java内置的spi通过java.util.serviceloader类解析classpath和jar包的meta-inf/services/目录 下的以接口全限定名命名的文件,并加载该文件中指定的接口实现类,以此完成调用。
示例说明
创建动态接口
public interface vediospi{ void call();}
实现类1
public class mp3vedio implements vediospi{ @override public void call() { system.out.println(this is mp3 call); }}
实现类2
public class mp4vedio implements vediospi{ @override public void call() { system.out.println(this is mp4 call); }}
在项目的source目录下新建meta-inf/services/目录下,创建com.skywares.fw.juc.spi.vediospi文件。
相关测试
public class vediospitest{ public static void main(string[] args) { serviceloader serviceloader =serviceloader.load(vediospi.class); serviceloader.foreach(t->{ t.call(); }); }}
说明:java实现spi是通过serviceloader来查找服务提供的工具类。
运行结果:
源码分析
上述只是通过简单的示例来实现下java的内置的spi功能。其实现原理是serviceloader是java内置的用于查找服务提供接口的工具类,通过调用load()方法实现对服务提供接口的查找,最后遍历来逐个访问服务提供接口的实现类。
从源码可以发现:
serviceloader类本身实现了iterable接口并实现了其中的iterator方法,iterator方法的实现中调用了lazyiterator这个内部类中的方法,迭代器创建实例。
所有服务提供接口的对应文件都是放置在meta-inf/services/目录下,final类型决定了prefix目录不可变更。
虽然java提供的spi机制的思想非常好,但是也存在相应的弊端。具体如下:
java内置的方法方式只能通过遍历来获取
服务提供接口必须放到meta-inf/services/目录下。
针对java的spi存在的问题,spring的spi机制沿用的spi的思想,但对其进行扩展和优化。
spring spi
spring spi沿用了java spi的设计思想,spring采用的是spring.factories方式实现spi机制,可以在不修改spring源码的前提下,提供spring框架的扩展性。
spring 示例
定义接口
public interface databasespi{ void getconnection();}
相关实现
##db2实现public class db2database implements databasespi{ @override public void getconnection() { system.out.println(this database is db2); }}##mysql实现public class mysqldatabase implements databasespi{ @override public void getconnection() { system.out.println(this is mysql database); }}
1、在项目的meta-inf目录下,新增spring.factories文件
2、填写相关的接口信息,内容如下:
com.skywares.fw.juc.springspi.databasespi = com.skywares.fw.juc.springspi.db2database, com.skywares.fw.juc.springspi.mysqldatabase
说明多个实现采用逗号分隔。
相关测试类
public class springspitest{ public static void main(string[] args) { list databasespis =springfactoriesloader.loadfactories(databasespi.class, thread.currentthread().getcontextclassloader()); for(databasespi datbasespi:databasespis){ datbasespi.getconnection(); } }}
输出结果
从示例中我们看出,spring 采用spring.factories实现spi与java实现spi非常相似,但是spring的spi方式针对java的spi进行的相关优化具体内容如下:
java spi是一个服务提供接口对应一个配置文件,配置文件中存放当前接口的所有实现类,多个服务提供接口对应多个配置文件,所有配置都在services目录下;
spring factories spi是一个spring.factories配置文件存放多个接口及对应的实现类,以接口全限定名作为key,实现类作为value来配置,多个实现类用逗号隔开,仅spring.factories一个配置文件。
那么spring是如何通过加载spring.factories来实现spi的呢?我们可以通过源码来进一步分析。
源码分析
说明:loadfactorynames解析spring.factories文件中指定接口的实现类的全限定名,具体实现如下:
说明:获取所有jar包中meta-inf/spring.factories文件路径,以枚举值返回。遍历spring.factories文件路径,逐个加载解析,整合factoryclass类型的实现类名称,获取到实现类的全类名称后进行类的实例话操作,其相关源码如下:
说明:实例化是通过反射来实现对应的初始化。
基于arm cortex m0的MM32SPIN05TW之无传感方波驱动水泵方案
【节能学院】浅谈光伏运维平台在机场项目的应用和效益
太阳能逆变器的应用与发展前景
NXP NXH3670UK芯片的功能特点及应用解决方案
荣耀9怎么样?华为荣耀9评测:荣耀9闪瞎眼,华为荣耀8持续降价只为荣耀9
JDK内置的一种服务SPI机制
这个精准又稳健的组合是如何满足恶劣环境下的输出要求的?
利用LoRaWAN网关内置的Server来组网的两种方式的基本框架
“白菜价”的iPhone8你会不会买单?
随着5G产业的不断发展,虚拟现实领域也将迎来新生
干法刻蚀原理
华为在多项5G-A上下行超宽带技术上取得重大性能突破
动力电池原理及类型
各国政府参与太空竞赛 争夺几万亿美元大蛋糕
选择陶瓷电容为什么要看厂家生产资质呢?
Orange发力物联网市场,通过增值服务方式收集和输入数据
蓄电池充电注意事项
AFPM100消防设备电源监控系统在国航重庆分公司维修基地项目中的应用
铝基板多少钱一平米_铝基板结构与优势
众矢之的!OPPO与VIVO自小米后将成被围殴的对象