Node.js工程师养成打算完结含文档源码
点击下载:网盘链接
深入剖析Spring Boot 的SPI机制
简介
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
{
@Overridepublic void call(){ System.out.println("this is mp3 call");}
}
复制代码
实现类2
public class Mp4Vedio implements VedioSPI
{
@Overridepublic 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 SPI
Spring SPI沿用了Java SPI的设计思维,Spring采纳的是spring.factories形式实现SPI机制,可能在不修改Spring源码的前提下,提供Spring框架的扩展性。
Spring 示例
定义接口
public interface DataBaseSPI
{
void getConnection();
}
复制代码
相干实现
DB2实现
public class DB2DataBase implements DataBaseSPI
{
@Overridepublic void getConnection(){ System.out.println("this database is db2");}
}
Mysql实现
public class MysqlDataBase implements DataBaseSPI
{
@Overridepublic 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一个配置文件。
总结
本文粗疏的讲解了java和Spring的SPI机制,SPI技术将服务接口与服务实现进行分离实现解耦,从而晋升程序的可扩展性。