0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

JDK内置的一种服务SPI机制

jf_ro2CN3Fa 来源:稀土掘金 2023-02-15 09:15 次阅读

SPI(Service Provider Interface)是JDK内置的一种服务提供发现机制,可以用来启用框架扩展和替换组件,主要用于框架中开发,例如Dubbo、Spring、Common-Logging,JDBC等采用采用SPI机制,针对同一接口采用不同的实现提供给不同的用户,从而提高了框架的扩展性。

Java SPI实现

Java内置的SPI通过java.util.ServiceLoader类解析classPath和jar包的META-INF/services/目录 下的以接口全限定名命名的文件,并加载该文件中指定的接口实现类,以此完成调用。

示例说明

创建动态接口

publicinterfaceVedioSPI
{
voidcall();
}

实现类1

publicclassMp3VedioimplementsVedioSPI
{
@Override
publicvoidcall()
{
System.out.println("thisismp3call");
}

}

实现类2

publicclassMp4VedioimplementsVedioSPI
{
@Override
publicvoidcall()
{
System.out.println("thisismp4call");
}

}

在项目的source目录下新建META-INF/services/目录下,创建com.skywares.fw.juc.spi.VedioSPI文件。

2dc32b86-ac5d-11ed-bfe3-dac502259ad0.png

相关测试

publicclassVedioSPITest
{
publicstaticvoidmain(String[]args)
{
ServiceLoaderserviceLoader=ServiceLoader.load(VedioSPI.class);

serviceLoader.forEach(t->{
t.call();
});
}
}

说明:Java实现spi是通过ServiceLoader来查找服务提供的工具类。

运行结果:

2dd556f8-ac5d-11ed-bfe3-dac502259ad0.png

源码分析

上述只是通过简单的示例来实现下java的内置的SPI功能。其实现原理是ServiceLoader是Java内置的用于查找服务提供接口的工具类,通过调用load()方法实现对服务提供接口的查找,最后遍历来逐个访问服务提供接口的实现类。

2de67e92-ac5d-11ed-bfe3-dac502259ad0.png

从源码可以发现:

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 示例

定义接口

publicinterfaceDataBaseSPI
{
voidgetConnection();
}

相关实现

##DB2实现
publicclassDB2DataBaseimplementsDataBaseSPI
{
@Override
publicvoidgetConnection()
{
System.out.println("thisdatabaseisdb2");
}

}

##Mysql实现
publicclassMysqlDataBaseimplementsDataBaseSPI
{
@Override
publicvoidgetConnection()
{
System.out.println("thisismysqldatabase");
}

}

1、在项目的META-INF目录下,新增spring.factories文件

2df71af4-ac5d-11ed-bfe3-dac502259ad0.png

2、填写相关的接口信息,内容如下:

com.skywares.fw.juc.springspi.DataBaseSPI=com.skywares.fw.juc.springspi.DB2DataBase,com.skywares.fw.juc.springspi.MysqlDataBase

说明多个实现采用逗号分隔。

相关测试类

publicclassSpringSPITest
{
publicstaticvoidmain(String[]args)
{
ListdataBaseSPIs=SpringFactoriesLoader.loadFactories(DataBaseSPI.class,
Thread.currentThread().getContextClassLoader());

for(DataBaseSPIdatBaseSPI:dataBaseSPIs){
datBaseSPI.getConnection();
}
}
}

输出结果

2e061d9c-ac5d-11ed-bfe3-dac502259ad0.png

从示例中我们看出,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的呢?我们可以通过源码来进一步分析。

源码分析

2e1b1332-ac5d-11ed-bfe3-dac502259ad0.png

说明:loadFactoryNames解析spring.factories文件中指定接口的实现类的全限定名,具体实现如下:

2e312384-ac5d-11ed-bfe3-dac502259ad0.png

说明:获取所有jar包中META-INF/spring.factories文件路径,以枚举值返回。遍历spring.factories文件路径,逐个加载解析,整合factoryClass类型的实现类名称,获取到实现类的全类名称后进行类的实例话操作,其相关源码如下:

2e4bdbb6-ac5d-11ed-bfe3-dac502259ad0.png

说明:实例化是通过反射来实现对应的初始化。







审核编辑:刘清

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • JAVA
    +关注

    关注

    19

    文章

    2904

    浏览量

    103003
  • SPI
    SPI
    +关注

    关注

    17

    文章

    1615

    浏览量

    89674
  • JDBC
    +关注

    关注

    0

    文章

    25

    浏览量

    13299

原文标题:深入剖析 Spring Boot 的 SPI 机制

文章出处:【微信号:芋道源码,微信公众号:芋道源码】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    一种用单片机制作的高频正弦波逆变器

    一种用单片机制作的高频正弦波逆变器
    发表于 04-08 12:40

    请问SYS/BIOS是否提供了一种多核内存管理的机制

    在使用heapMem模块时发现该模块只是提供的单核的内存管理,请问SYS/BIOS是否提供了一种多核内存管理的机制
    发表于 01-04 11:40

    请问板子的延迟函数是一种什么机制

    我想请问下,你们板子的延迟函数是一种什么机制,如以上的定义所示,他的延时单位是ms还是其他,就像Delay(0x000FFFFF)算是延时了多少秒?有没有相关的资料视屏什么的讲解
    发表于 10-15 10:40

    求大神分享一种基于bootloader的嵌入式软件自动更新机制

    本文提出了一种具有较高稳定性和安全性、基于bootloader的嵌入式软件自动更新机制。该更新机制同时保存了3个文件,需要较多的Flash存储空间,但同时降低了维护成本。
    发表于 04-27 06:33

    请问怎样去设计一种SPI接口?

    请问怎样去设计一种SPI接口?
    发表于 04-28 06:50

    请教大神如何去设计一种SPI4.2接口?

    本文介绍了一种FPGA和IPX2805之间的SPI4.2接口模块设计的方法,对硬件设计进行了说明,着重阐述了FPGA内部SPI4.2接口模块设计。
    发表于 05-06 09:22

    怎样去设计一种同步补偿机制(CDCO算法)?

    本文基于发送方接收方双向同步算法的原理,提出一种基于时钟漂移与偏移的同步补偿机制(CDCO算法)。
    发表于 05-17 06:18

    分享一种高性能的FM内置天线解决方案

    分享一种高性能的FM内置天线解决方案
    发表于 05-26 06:18

    如何去实现一种SPI接口电路?

    SPI总线协议是什么?如何去实现一种SPI接口电路?
    发表于 05-28 07:08

    怎样去设计一种采用覆盖机制的FIFO队列模型呢

    FIFO队列是什么?怎样去设计一种采用覆盖机制的FIFO队列模型呢?
    发表于 12-08 06:07

    详解一种同步全双工串行接口SPI

    SPI英文全称(Serial Perripheral Interface),即串行外围设备接口,是一种同步全双工串行接口,MCU可以通过SPI方式与各种外围设备进行高速数据通信
    发表于 02-17 07:07

    如何在Ota服务器上安装JDK版本呢

    如何在Ota服务器上安装JDK版本呢?具体安装步骤有哪些?
    发表于 03-03 12:15

    是否有一种机制可以从虚拟功能处理程序进行屏幕转换呢?

    我想知道是否有一种机制可以根据在滚动列表控件(小部件)上选择的项目转换到不同的屏幕,并传递有关所选项目或所选滚动列表项目的索引的信息,或者是否有一种机制可以从虚拟功能处理程序进行屏幕转
    发表于 12-13 08:09

    可插拔组件设计机制SPI介绍

    SPI 的全称是 Service Provider Interface, 即提供服务接口;是一种服务发现机制
    的头像 发表于 03-23 09:20 828次阅读

    什么是SPI机制

    的ContextClassLoader加载以便使用)。本次将对 SPI机制进行详解,并结合案例介绍其在实际场景中具体使用。 2、什么是SPI机制
    的头像 发表于 10-08 15:03 570次阅读
    什么是<b class='flag-5'>SPI</b><b class='flag-5'>机制</b>