深入剖析 Spring Boot 的 SPI 机制

关注我,回复关键字“spring”免费领取Spring学习资料来源:juejin.cn/post/7132742686099898398SPI(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<VedioSPI> 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<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类型的实现类名称,获取到实现类的全类名称后进行类的实例话操作,其相关源码如下:图片说明:实例化是通过反射来实现对应的初始化。

END



阿里版ChatGPT已进入测试,中文聊天截图曝光!
从历代GC算法角度刨析ZGC
再见 Feign ,Spring 6 新特性:HTTP Interface 来了!
终于搞懂了 @Configuration 和 @Component 的区别 !

关注后端面试那些事,回复【2022面经】

获取最新大厂Java面经

最后重要提示:高质量的技术交流群,限时免费开放,今年抱团最重要。想进群的,关注SpringForAll社区,回复关键词:加群,拉你进群。
点击这里领取2022大厂面经

相关推荐

  • 微信剧变
  • 这是我见过最好的OA系统!
  • 系统上线前,SQL脚本的9大坑
  • 京东一面:MySQL 中的 distinct 和 group by 哪个效率更高?
  • ChatGPT成功做对了这4步丨爱丁堡大学符尧
  • ChatGPT数据集之谜
  • 张家俊:关于ChatGPT八个技术问题的猜想
  • 多线程环境下,HashMap为什么会出现死循环?
  • 手写一个性能较好的拖拽排序(上)
  • Prometheus + Grafana:一套 SpringBoot 应用的可视化监控方案!
  • 高度近视父母1岁孩子近视600度;送餐小哥揪出113名吸贩毒人员;机器人液化后“越狱”;苏州一地火灾致6死28伤...|酷玩日爆
  • 我承认这次是美国赢了! | 每日一冷
  • 日本成立亚洲第一支太空军,到底想干什么?
  • 2022年,超5700家中国芯片公司消失
  • 一天18个小时离不开电子屏幕,谁来救救我的眼睛?
  • 中国风的“西餐叛徒”,要来整顿国内快餐行业了?
  • 北大校长的忠告火了!孩子14岁前,家长千万别在这件事上偷懒
  • 十几万库存!价格压至极低:超硬核的冲锋衣,99起
  • 韩检方提请批捕李在明;苏格兰“总统”斯特金宣布辞职;“毒气列车”事件受害居民讨说法,铁路公司因害怕未出面 | 每日大新闻
  • 联邦GNN综述与经典算法介绍