SpringBoot SPI机制理解

简介: SpringBoot SPI机制理解

什么是SPI

SPI的全称是Service Provider Interface,服务提供接口。

简单来说就是通过配置文件指定接口的实现类

当我们开发一套框架,一套机制,一个插件,或者一套API的时候,如果需要第三方的服务支持,可以直接写死在代码里面,但是这种方式耦合太强,不利于切换到其他服务,好的方式就是指定一个配置文件,指定服务的实现方。jdk的spi就是这种机制。

一个接口可以有很多实现,比如数据库驱动,有oracle,mysql,postgress等等,他们都遵循JDBC规范,为了解耦,我们可以抽象出一个高层的Driver接口,让各个数据库服务商去实现各自的驱动,在使用的时候我们可以选择加载具体的实现方式,这时候我们就可以使用SPI这种技术。

JDK中的SPI

讲到JDK中的SPI ,我们不得不说 java.util.ServiceLoader这个类,我们先跑起来

  1. 创建一个接口,Message

    public interface Message{
        void send()
    }
  2. 在resources资源目录下创建META-INF/services文件夹
  3. 在services文件夹中创建文件,以接口全名命名
  4. 创建接口实现类

    public class SmsMessage implements Message{
        public void send(){
            System.out.println("send sms message");
        }
    }
    
    public class EmailMessage implements Message{   
        public void send(){
            System.out.println("send email message");
        }
    }
  5. 测试

    public class TestServiceLoader{
        public static void main(String[] args){
            ServiceLoader<Message> messages = ServiceLoader.load(Message.class);
            for(Message msg : messages){
                msg.send();
            }
        }
    }

SpringBoot中的SPI

在SpringBoot的自动装配过程中,最终会加载META-INF/spring.factories文件,SpringBoot是通过SpringFactoriesLoader#loadFactoryNames方法加载的。从classpath下的每一个jar包中搜寻所有META-INF/spring.factories配置文件,然后将解析properties文件,找到指定名称的配置后返回。需要注意的是,其实这里不仅仅是会去ClassPath路径下查找,会扫描所有路径下的Jar包,只不过这个文件只会在Classpath下的jar包中。

public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) {
        ClassLoader classLoaderToUse = classLoader;
        if (classLoaderToUse == null) {
            classLoaderToUse = SpringFactoriesLoader.class.getClassLoader();
        }
        String factoryTypeName = factoryType.getName();
        return loadSpringFactories(classLoaderToUse).getOrDefault(factoryTypeName, Collections.emptyList());
}

private static Map<String, List<String>> loadSpringFactories(ClassLoader classLoader) {
        Map<String, List<String>> result = cache.get(classLoader);
        if (result != null) {
            return result;
        }

        result = new HashMap<>();
        try {
            Enumeration<URL> urls = classLoader.getResources(FACTORIES_RESOURCE_LOCATION);
            while (urls.hasMoreElements()) {
                URL url = urls.nextElement();
                UrlResource resource = new UrlResource(url);
                Properties properties = PropertiesLoaderUtils.loadProperties(resource);
                for (Map.Entry<?, ?> entry : properties.entrySet()) {
                    String factoryTypeName = ((String) entry.getKey()).trim();
                    String[] factoryImplementationNames =
                            StringUtils.commaDelimitedListToStringArray((String) entry.getValue());
                    for (String factoryImplementationName : factoryImplementationNames) {
                        result.computeIfAbsent(factoryTypeName, key -> new ArrayList<>())
                                .add(factoryImplementationName.trim());
                    }
                }
            }

            // Replace all lists with unmodifiable lists containing unique elements
            result.replaceAll((factoryType, implementations) -> implementations.stream().distinct()
                    .collect(Collectors.collectingAndThen(Collectors.toList(), Collections::unmodifiableList)));
            cache.put(classLoader, result);
        }
        catch (IOException ex) {
            throw new IllegalArgumentException("Unable to load factories from location [" +
                    FACTORIES_RESOURCE_LOCATION + "]", ex);
        }
        return result;
}

SPI使用场景

适合根据实际使用,更换实现策略的框架。

如下:

  1. JDBC负载驱动不同类型的数据库。
  2. SLF4J载入不同供应商的日志实现类别。

    等等

相关文章
|
1月前
|
JavaScript 前端开发 Java
基于SpringBoot+Vue实现前后端交互功能(详解Vue框架机制)
基于SpringBoot+Vue实现前后端交互功能(详解Vue框架机制)
|
1月前
|
Java 开发者 UED
Spring Boot的全局异常处理机制
【2月更文挑战第13天】
120 0
|
1月前
|
人工智能 JSON 前端开发
【Spring boot实战】Springboot+对话ai模型整体框架+高并发线程机制处理优化+提示词工程效果展示(按照框架自己修改可对接市面上百分之99的模型)
【Spring boot实战】Springboot+对话ai模型整体框架+高并发线程机制处理优化+提示词工程效果展示(按照框架自己修改可对接市面上百分之99的模型)
|
1月前
|
设计模式 Java 机器人
SpringBoot3自动配置流程 SPI机制 核心注解 自定义starter
SpringBoot3自动配置流程 SPI机制 核心注解 自定义starter
|
7天前
|
安全 Java 应用服务中间件
优雅的理解SpringBoot的核心机制
【6月更文挑战第21天】理解 Spring Boot 的核心机制可以帮助开发者更好地利用这个框架来构建现代化的企业级应用。
25 9
|
9天前
|
Java 应用服务中间件 Spring
解析Spring Boot自动装配的原理与机制
解析Spring Boot自动装配的原理与机制
19 4
|
1天前
|
Java Spring 容器
在Spring Boot中实现类似SPI机制的功能(二)
在Spring Boot中实现类似SPI机制的功能(二)
9 0
|
1月前
|
XML Java 开发者
springboot 启动原理、启动过程、启动机制的介绍
【5月更文挑战第13天】Spring Boot 是一种基于 Java 的框架,用于创建独立的、生产级别的 Spring 应用程序。它的主要目标是简化 Spring 应用的初始搭建和开发过程,同时提供一系列大型项目常见的非功能性特征(如嵌入式服务器、安全性、度量、健康检查和外部化配置)。
44 3
|
1月前
|
JSON 前端开发 Java
深入解析SpringBoot的请求响应机制
深入解析SpringBoot的请求响应机制
|
13小时前
|
JavaScript Java 测试技术
基于SpringBoot+Vue的宠物救助管理系统的详细设计和实现(源码+lw+部署文档+讲解等)
基于SpringBoot+Vue的宠物救助管理系统的详细设计和实现(源码+lw+部署文档+讲解等)
11 0