深入剖析 Spring Boot 的 SPI 机制

点击关注公众号:互联网架构师,后台回复 2T获取2TB学习资源!

上一篇:Alibaba开源内网高并发编程手册.pdf

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

之前也写过Java SPI的深入剖析:聊聊 Java 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文件。

892c427ddb4717970e0920ed091cc95c.jpeg

相关测试

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

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

运行结果:

64399ee6a0d5591b211fee1cf36d3a46.jpeg

源码分析

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

5c1524e964134a1aaf91ad902137a279.jpeg

从源码可以发现:

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

b3623fb8d2b0b8aadf55ec68da889c41.png

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();
         }
    }
}

输出结果

74466e972e03f1fa8f73b9edb942fcb1.jpeg

从示例中我们看出,Spring 采用spring.factories实现SPI与java实现SPI非常相似,但是spring的spi方式针对java的spi进行的相关优化具体内容如下:

  • Java SPI是一个服务提供接口对应一个配置文件,配置文件中存放当前接口的所有实现类,多个服务提供接口对应多个配置文件,所有配置都在services目录下;关注公众号:互联网架构师,回复关键词:9 获取阿里内部Java性能调优手册!

  • Spring factories SPI是一个spring.factories配置文件存放多个接口及对应的实现类,以接口全限定名作为key,实现类作为value来配置,多个实现类用逗号隔开,仅spring.factories一个配置文件。

那么spring是如何通过加载spring.factories来实现SpI的呢?我们可以通过源码来进一步分析。

源码分析

6c2f6b50e503903603f43d56c3cf596b.jpeg

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

adc3ae9f40836ef0112bdd9f285fda84.jpeg

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

7f399e8be3b806bf82eef8077d2239ad.jpeg

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

https://juejin.cn/post/7132742686099898398

最后,关注公众号互联网架构师,在后台回复:2T,可以获取我整理的 Java 系列面试题和答案,非常齐全。

正文结束

推荐阅读 ↓↓↓

1.CTO:谁在项目中使用Arrays.asList、ArrayList.subList,就立马滚蛋!

2.Alibaba开源内网高并发编程手册.pdf

3.程序员一般可以从什么平台接私活?

4.40岁,刚被裁,想说点啥。

5.为什么国内 996 干不过国外的 955呢?

6.中国的铁路订票系统在世界上属于什么水平?                        

7.15张图看懂瞎忙和高效的区别!

814edb2e8d36513fe8714712fec60cb1.gif

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot中的SPI(Service Provider Interface)机制可以帮助我们扩展应用程序的功能。SPI机制Java提供的一种基于接口的编程规范,允许第三方服务提供者来扩展应用程序的功能。在Spring Boot中,我们可以使用SPI机制来实现插件化的开发。 具体来说,Spring Boot中的SPI机制包括以下几个步骤: 1. 定义接口:我们需要定义一个接口,用于描述我们要扩展的功能。 2. 实现接口:第三方服务提供者需要实现这个接口,并将实现类打成JAR包。 3. 在META-INF/services目录下创建文件:我们需要在项目的META-INF/services目录下创建一个文件,文件名为接口的全限定名,文件内容为实现类的全限定名。 4. 使用扩展点:当应用程序启动时,Spring Boot会自动扫描META-INF/services目录下的文件,并加载实现类,我们可以通过Spring容器获取接口的实现类对象,从而使用扩展点提供的功能。 下面是一个简单的示例,演示了如何使用Spring Boot中的SPI机制: 1. 定义接口: ```java public interface GreetingService { String greet(String name); } ``` 2. 实现接口: ```java public class GreetingServiceImpl implements GreetingService { @Override public String greet(String name) { return "Hello, " + name + "!"; } } ``` 3. 在META-INF/services目录下创建文件: 创建一个名为“com.example.demo.GreetingService”的文件,文件内容为: ``` com.example.demo.GreetingServiceImpl ``` 4. 使用扩展点: ```java @Service public class MyService { @Autowired private List<GreetingService> greetingServices; public void greet(String name) { for (GreetingService greetingService : greetingServices) { System.out.println(greetingService.greet(name)); } } } ``` 在上面的示例中,我们通过@Autowired注解注入了一个List<GreetingService>类型的对象,这个对象包含了所有实现了GreetingService接口的类。我们可以通过遍历这个List对象,依次调用每个实现类的greet方法,从而实现了插件化的开发。 总之,Spring Boot中的SPI机制可以帮助我们扩展应用程序的功能,提高代码的可扩展性和可维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值