java spi实现java内置的spi通过java.util.serviceloader类解析classpath和jar包的meta-inf/services/目录 下的以接口全限定名命名的文件,并加载该文件中指定的接口实现类,以此完成调用。
示例说明创建动态接口public interface vediospi{ void call();}
实现类1public class mp3vedio implements vediospi{ @override public void call() { system.out.println("this is mp3 call"); }}
实现类2public 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<vediospi> serviceloader =serviceloader.load(vediospi.class); serviceloader.foreach(t->{ t.call(); }); }}
说明:java实现spi是通过serviceloader来查找服务提供的工具类。
运行结果
源码分析上述只是通过简单的示例来实现下java的内置的spi功能。其实现原理是serviceloader是java内置的用于查找服务提供接口的工具类,通过调用load()方法实现对服务提供接口的查找,最后遍历来逐个访问服务提供接口的实现类。
从源码可以发现:
serviceloader类本身实现了iterable接口并实现了其中的iterator方法,iterator方法的实现中调用了lazyiterator这个内部类中的方法,解析完服务提供接口文件后最终结果放在了iterator中返回,并不支持服务提供接口实现类的直接访问。
所有服务提供接口的对应文件都是放置在meta-inf/services/目录下,final类型决定了prefix目录不可变更。
虽然java提供的spi机制的思想非常好,但是也存在相应的弊端。具体如下:
java内置的方法方式只能通过遍历来获取
服务提供接口必须放到meta-inf/services/目录下。
针对java的spi存在的问题,spring的spi机制沿用的spi的思想,但对其进行扩展和优化。
spring spispring 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<databasespi> 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类型的实现类名称,获取到实现类的全类名称后进行类的实例话操作,其相关源码如下:
说明:实例化是通过反射来实现对应的初始化。
以上就是springboot的spi机制怎么实现的详细内容。
