Java-SPI应用场景以及原理

一、前言(介绍)

Java SPI(Service Provider Interface)是 Java 官方提供的一种服务发现机制 ,它允许在运行时动态的加载实现特定接口的类

通过定义接口,实现类去具体实现,然后使用方根据自己项目扫描出来的实现类来调用(依赖 jar),实现可插拔式的开发。

二、基础 Demo

1、定义 Service spi

csharp 复制代码
public interface CarApi {
    String builderCar();
}

2、实现 Service Provider

typescript 复制代码
public class MiCarServiceImpl implements CarApi {
    @Override
    public String builderCar() {
        return "小米电车su7.。。。。。。。。。。。。。。。。。";
    }
}

3、在 Serivice Provider 的 resources目录下,创建META-INF/services目录

4、服务发现

vbnet 复制代码
1、注入依赖(Service Provider 的jar包)
2、调用服务
public void testCar(){
        ServiceLoader<CarApi> carApis = ServiceLoader.load(CarApi.class);
        Iterator<CarApi> iterator = carApis.iterator();
        while (iterator.hasNext()){
            CarApi next = iterator.next();
            System.out.println(next.builderCar());
        }
}

代码地址:gitee.com/boge1022/ja...

三、实现原理

工作流程:

角色分类:

  • Service Api :是一个公开的接口或者抽象类,定义了一个抽象的功能模块
  • Serivice Provider:是 Service api 接口的一个实现类
  • ServiceLoader:是 SPI 机制中核心组件,负责在运行时发现并加载 Service Provider

Service Provider 规范要素:

文件路径:必须在 jar 包中的 META-INF/services 目录下

文件名称:Service Api 接口全限类名

文件内容:Service Provider 实现类全限类名,如果有多个实现类,那么每一个实现类单独占用一行

注意:Service Provider 实现类必须具备无参的构造方法。因为后面是通过反射技术创建对象

原理:

SPI 加载器:使用线程上下文加载器,默认是 ApplicationClassloader 加载器。

通过固定格式 META-INF/service+Service Api 接口全限类名 来使用 inputStream 流读取文件信息,拿到实现类的全限类名,最后通过反射的方式实例化对象,放入缓存中

四、应用场景(待补充)

通常用于一个方法多个实现,但具体实现内容选取取决于依赖的服务提供者。

1、日志框架slf4j

2、DriverManager 驱动选取策略

3、sharding-jdbc 主键生成策略

五、总结

Java 中的 SPI 提供一种特殊的服务发现机制,通过接口将服务调用和服务提供分隔开,将接口提供给第三方实现扩展,体现了依赖倒置的设计思想,代码解藕。

个人总结:Java 中的 SPI 的理解

1、从思想层面来理解:SPI 提供了一种特殊的服务发现机制,类似于策略模式的思想,通过依赖的提供者来调用具体的实现类

2、从技术层面来说:主要是将服务调用者和服务提供者分隔开,通过服务发现来获取具体的提供者实现(jar 包**)**

相关推荐
小鱼人爱编程1 小时前
Java基石--反射让你直捣黄龙
前端·spring boot·后端
hqxstudying1 小时前
J2EE模式---服务层模式
java·数据库·后端·spring·oracle·java-ee
GM_8281 小时前
【最新最完整】SpringAI-1.0.0开发MCP Server,搭建MCP Client 实战笔记(进阶+详细+完整代码)
java·后端·ai编程·springai·mcp
程序员爱钓鱼1 小时前
Go语言实战案例-滑动窗口最大值
后端·google·go
Victor3562 小时前
MySQL(163) 如何理解MySQL的隔离级别?
后端
Victor3562 小时前
MySQL(164)如何设置MySQL的隔离级别?
后端
代码老y3 小时前
ASP.NET Core 高并发万字攻防战:架构设计、性能优化与生产实践
后端·性能优化·asp.net
武子康8 小时前
Java-80 深入浅出 RPC Dubbo 动态服务降级:从雪崩防护到配置中心秒级生效
java·分布式·后端·spring·微服务·rpc·dubbo
舒一笑9 小时前
我的开源项目-PandaCoder迎来史诗级大更新啦
后端·程序员·intellij idea
@昵称不存在10 小时前
Flask input 和datalist结合
后端·python·flask