携手创作,一起成长!这是我参加「日新方案 8 月更文应战」的第21天,点击检查活动概况

简介

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文件。

深入剖析Spring Boot 的SPI机制

相关测验

public class VedioSPITest
{
    public static void main(String[] args)
    {
        ServiceLoader<VedioSPI> serviceLoader =ServiceLoader.load(VedioSPI.class);
        serviceLoader.forEach(t->{
            t.call();
        });
    }
}

阐明:Java完结spi是经过ServiceLoader来查找服务供给的工具类。

运转成果:

深入剖析Spring Boot 的SPI机制

源码剖析

上述仅仅经过简略的示例来完结下java的内置的SPI功用。其完结原理是ServiceLoader是Java内置的用于查找服务供给接口的工具类,经过调用load()办法完结对服务供给接口的查找,最终遍历来逐个访问服务供给接口的完结类。

深入剖析Spring Boot 的SPI机制

从源码能够发现:

  • 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文件

深入剖析Spring Boot 的SPI机制

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 Boot 的SPI机制

从示例中咱们看出,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的呢?咱们能够经过源码来进一步剖析。

源码剖析

深入剖析Spring Boot 的SPI机制

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

深入剖析Spring Boot 的SPI机制
阐明: 获取一切jar包中META-INF/spring.factories文件途径,以枚举值返回。 遍历spring.factories文件途径,逐个加载解析,整合factoryClass类型的完结类称号,获取到完结类的全类称号后进行类的实例话操作,其相关源码如下:

深入剖析Spring Boot 的SPI机制

阐明:实例化是经过反射来完结对应的初始化

总结

本文具体的讲解了java和Spring的SPI机制,SPI技术将服务接口与服务完结进行别离完结解耦,从而提高程序的可扩展性。如有疑问,请随时反馈。