SPI机制
一、SPI简介
SPI(Service Provider Interface)机制是一种服务发现机制,广泛用于Java生态中。它允许框架或库通过接口解耦具体实现,用户可以在运行时动态地提供接口的实现,而不是在编译时确定。这种机制在很多场景下非常有用,比如数据库驱动、日志框架、解析库等。
二、SPI原理
SPI机制的核心思想是,接口的实现类通过配置文件进行声明,而框架或服务的调用方在运行时可以动态加载这些实现类。
1. 接口定义
首先,开发者定义一个接口(或抽象类),该接口规定了需要实现的功能。例如,一个简单的接口可以是这样:
public interface MyService {void execute();
}
2. 服务提供者(实现接口的类)
接下来,不同的提供者可以实现这个接口,提供不同的功能。例如,提供者A和提供者B分别实现MyService接口:
public class MyServiceProviderA implements MyService {@Overridepublic void execute() {System.out.println("MyServiceProviderA execution");}
}public class MyServiceProviderB implements MyService {@Overridepublic void execute() {System.out.println("MyServiceProviderB execution");}
}
3. 定义服务提供者配置文件
SPI机制依赖于META-INF/services目录下的配置文件。配置文件的名称必须是接口的全限定名,内容是接口实现类的全限定名。对于MyService接口,其配置文件应该命名为:
META-INF/services/com.example.MyService
该文件内容如下,包含每个实现类的全限定类名:
com.example.MyServiceProviderA
com.example.MyServiceProviderB
4. 使用ServiceLoader动态加载服务
Java提供了ServiceLoader类来查找和加载服务。ServiceLoader会扫描META-INF/services目录,并加载该接口的所有实现类:
ServiceLoader<MyService> serviceLoader = ServiceLoader.load(MyService.class);
for (MyService service : serviceLoader) {service.execute();
}
ServiceLoader会自动查找META-INF/services/com.example.MyService文件,并实例化该文件中定义的所有类。每个实现类的execute方法都会被调用。
三、SPI的优劣
SPI的优势
- 解耦性:通过接口来解耦,实现类可以动态加载,调用方不需要依赖具体实现。
- 扩展性强:可以很方便地通过提供不同的实现类来扩展功能,不需要修改框架代码。
- 动态加载:SPI机制允许在运行时动态选择或加载实现类,这非常适合插件化的框架。
SPI的局限性
尽管SPI提供了很大的灵活性,但也有一些缺点:
- 无法选择实现:
ServiceLoader会加载所有提供者,如果你有多个实现类,你无法直接选择使用哪一个,除非通过额外的逻辑来筛选。 - 性能问题:每次调用
ServiceLoader.load都会遍历META-INF/services目录并加载类,这对性能可能有一定的影响。 - 安全性问题:加载外部服务实现时,如果实现类存在安全漏洞或者不安全的代码,可能导致问题。
SPI的典型应用场景
- JDBC:JDBC是SPI的典型应用。JDBC接口定义了数据库操作的规范,不同的数据库驱动程序(如MySQL、Oracle等)通过实现JDBC接口为数据库提供访问功能。通过
META-INF/services文件,JDBC驱动在运行时可以被动态加载,无需硬编码到应用程序中。 - 日志框架:如SLF4J和Log4J,它们都提供了统一的接口,实际的日志实现可以是多种实现之一,比如Logback、Log4J等。框架可以在运行时根据配置文件加载不同的日志实现。
四、自定义SPI示例
1. 接口定义
public interface PaymentService {void pay(int amount);
}
2. 实现类
public class AlipayService implements PaymentService {@Overridepublic void pay(int amount) {System.out.println("Paid " + amount + " using Alipay");}
}public class WeChatPayService implements PaymentService {@Overridepublic void pay(int amount) {System.out.println("Paid " + amount + " using WeChat Pay");}
}
3. 配置文件
在META-INF/services目录下创建文件com.example.PaymentService,内容为:
com.example.AlipayService
com.example.WeChatPayService
4. 使用ServiceLoader加载并调用实现
ServiceLoader<PaymentService> services = ServiceLoader.load(PaymentService.class);
for (PaymentService service : services) {service.pay(100); // 调用所有服务提供者的pay方法
}
输出结果:
Paid 100 using Alipay
Paid 100 using WeChat Pay
总结
SPI机制为Java生态中的扩展和插件提供了标准的方式,尤其是在跨多个库或框架中非常有用。通过ServiceLoader和META-INF/services的配置,Java程序可以在运行时动态加载实现类,从而实现高扩展性和解耦。
五、Spring的伪SPI机制
Spring中的“伪SPI”机制,实际上是一种通过依赖注入和工厂模式实现的动态服务加载机制,和Java原生的SPI机制有一定的相似之处,但Spring通过容器管理的方式提供了更灵活、更强大的功能。
与Java原生的SPI机制相比,Spring的伪SPI机制不依赖ServiceLoader,而是通过Spring IoC容器的动态管理和加载实现类。这种机制不仅支持动态扩展,而且更具灵活性,因为它可以配合Spring的其他特性(如AOP、条件注解等)进行控制和优化。
Spring中的伪SPI机制实现方式
Spring提供了几种不同的方式实现类似SPI的动态加载机制,以下是一些常见的实现方式:
1. 通过@Conditional注解实现条件加载
Spring允许你根据不同的条件来动态加载Bean。这类似于SPI的选择性加载机制。
- 示例:
假设你有两个支付服务实现(类似前面的例子),你希望根据某些条件(如配置文件中的设置)来选择哪个服务被加载。
public interface PaymentService {void pay(int amount);
}public class AlipayService implements PaymentService {@Overridepublic void pay(int amount) {System.out.println("Paid " + amount + " using Alipay");}
}public class WeChatPayService implements PaymentService {@Overridepublic void pay(int amount) {System.out.println("Paid " + amount + " using WeChat Pay");}
}
你可以通过@Conditional注解动态选择加载哪一个支付服务实现:
- 基于配置文件的条件加载:
@Configuration
public class PaymentServiceConfig {@Bean@ConditionalOnProperty(name = "payment.service", havingValue = "alipay")public PaymentService alipayService() {return new AlipayService();}@Bean@ConditionalOnProperty(name = "payment.service", havingValue = "wechatpay")public PaymentService weChatPayService() {return new WeChatPayService();}
}
然后在application.properties中设置:
payment.service=alipay
这样,Spring容器在启动时会根据配置文件中的payment.service属性来决定加载哪个支付服务。
2. 使用@Primary或@Qualifier注解选择具体实现
当你有多个实现类时,Spring提供了选择加载哪个实现类的能力。使用@Primary可以设置默认的实现,而使用@Qualifier可以进行精确选择。
- 示例:
@Configuration
public class PaymentServiceConfig {@Bean@Primarypublic PaymentService alipayService() {return new AlipayService();}@Beanpublic PaymentService weChatPayService() {return new WeChatPayService();}
}
在使用时,你可以通过@Qualifier来精确指定使用哪一个实现:
@Service
public class PaymentProcessor {private final PaymentService paymentService;@Autowiredpublic PaymentProcessor(@Qualifier("weChatPayService") PaymentService paymentService) {this.paymentService = paymentService;}public void processPayment(int amount) {paymentService.pay(amount);}
}
这样,PaymentProcessor会使用weChatPayService作为具体的支付实现,而不是默认的alipayService。
3. 使用spring.factories文件的动态扩展
在Spring Boot中,类似于Java原生SPI的机制通过spring.factories文件来实现。这种方式通常用于自动配置类的加载,Spring Boot的自动配置依赖于它。
- 示例:
在META-INF/spring.factories文件中,可以定义哪些类会在应用启动时自动加载:
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.example.config.AlipayAutoConfiguration,\
com.example.config.WeChatPayAutoConfiguration
每当Spring Boot启动时,它会自动读取spring.factories文件并加载配置的类。这种方式允许开发者在不修改代码的情况下动态增加或替换实现,达到类似SPI的动态扩展效果。
4. 使用@AutoConfiguration机制
Spring Boot引入了自动配置机制,通过@AutoConfiguration和条件注解组合使用,使得自动装配更加灵活。可以通过依赖项的存在与否、配置属性等多种条件,动态决定是否装配某个实现。
- 示例:
@Configuration
@ConditionalOnClass(name = "com.example.payment.AlipayService")
public class AlipayAutoConfiguration {@Beanpublic PaymentService alipayService() {return new AlipayService();}
}
在这个例子中,只有在类路径上存在AlipayService类时,才会自动配置并创建alipayService的Bean。
5. 使用FactoryBean动态加载Bean
FactoryBean是Spring提供的一种特殊的Bean,它可以控制Bean的创建过程,允许根据逻辑条件返回不同的实现类。这为动态加载不同的服务提供了更多的灵活性。
- 示例:
public class PaymentServiceFactoryBean implements FactoryBean<PaymentService> {private String paymentType;public PaymentServiceFactoryBean(String paymentType) {this.paymentType = paymentType;}@Overridepublic PaymentService getObject() throws Exception {if ("alipay".equals(paymentType)) {return new AlipayService();} else {return new WeChatPayService();}}@Overridepublic Class<?> getObjectType() {return PaymentService.class;}@Overridepublic boolean isSingleton() {return true; // 返回true表示该Bean是单例的}
}
然后在配置类中使用:
@Configuration
public class PaymentConfig {@Beanpublic FactoryBean<PaymentService> paymentServiceFactoryBean() {return new PaymentServiceFactoryBean("alipay");}
}
这样,通过FactoryBean,你可以在运行时动态决定返回哪一个实现类。
六、Spring伪SPI机制与原生SPI的比较
| 特性 | 原生SPI | Spring伪SPI |
| 加载方式 |
|
, , |
| 配置文件 |
|
|
| 选择实现方式 | 自动加载所有实现,无法选择特定实现 | 通过 或 等灵活选择 |
| 扩展性 | 不支持条件加载,无法与其他特性集成 | 支持各种条件加载,灵活扩展 |
| 依赖注入支持 | 无依赖注入 | 支持依赖注入,且与Spring框架无缝集成 |
| 控制复杂性 | 配置简单,但功能相对单一 | 功能强大,可以结合多种条件、注解及工厂模式 |
总结
Spring的“伪SPI”机制通过依赖注入、条件注解、自动配置等机制,实现了比Java原生SPI更加灵活、强大的服务加载和扩展功能。它不仅能像SPI一样动态加载服务,还能根据各种条件灵活地控制加载行为,更适合复杂应用场景下的模块化和插件化开发。
这种伪SPI机制广泛应用于Spring Boot自动配置、扩展点设计和插件化架构中,极大地提升了Spring应用的扩展性和灵活性。
相关文章:
SPI机制
一、SPI简介 SPI(Service Provider Interface)机制是一种服务发现机制,广泛用于Java生态中。它允许框架或库通过接口解耦具体实现,用户可以在运行时动态地提供接口的实现,而不是在编译时确定。这种机制在很多场景下非…...
生信分析流程:从数据准备到结果解释的完整指南
介绍 生物信息学(生信)分析是一个复杂的过程,涉及从数据准备到结果解释的多个步骤。随着高通量测序技术的发展和生物数据的迅猛增长,了解和掌握生信分析的标准流程变得尤为重要。这不仅有助于提高分析的准确性,还能优…...
golang语法
参考链接:https://www.runoob.com/go/ 创建变量 // 3种方法 var a int a : 10 // 类型推断 a : make() // 复合类型循环 // 3种循环 for i : 0; i < 10; i {// 循环体} // 传统for循环 for index, num : range nums {// 循环体} // nums是可迭代的复合类型…...
【fisco学习记录2】多群组搭建
说明 文档参考: 多群组部署 — FISCO BCOS 2.0 v2.11.0 文档 (fisco-bcos-documentation.readthedocs.io) 多群组搭建之前,先暂停之前的单群组,并删除: cd fisco bash nodes/127.0.0.1/stop_all.sh rm -rf nodes/ 实现图&…...
深度解读:路由交换、负载均衡与防火墙的网络交响
一、路由交换:网络流动的“大动脉” 1. 路由:决定命运的“路径规划师” 路由技术如同现代交通网络中的导航系统,决定了数据从起点到终点的最佳路径。路由器基于网络层IP地址,对每个数据包进行精确的路径选择,并确保其…...
linux线程 | 线程的控制(二)
前言: 本节内容是线程的控制部分的第二个小节。 主要是列出我们的线程控制部分的几个细节性问题以及我们的线程分离。这些都是需要大量的代码去进行实验的。所以, 准备好接受新知识的友友们请耐心观看。 现在开始我们的学习吧。 ps:本节内容适合了解线程…...
npm install报错一堆sass gyp ERR!
执行npm install ,出现一堆gyp含有sass错误的情况下。 解决办法: 首页可能是node版本问题,太高或者太低,也会导致npm install安装错误(不会自动生成node_modules文件),本次试验,刚开…...
微知-BlueField DPU在lspci中显示Flash Recovery是什么意思?
效果: lspci |grep BlueField10:00.0 Memory controller: Mellanox Technologies MT42822 Family [BlueField-2 SoC Flash Recovery] (rev 01)*原因: 表示此时flash是empty空的,或者在flash中的FW是无法工作的。比如烧录错误。 这里指的一提…...
【前端知识点】前端笔记
css 引入css文件的文件路径 <!-- 引入外部 CSS 文件 --> <!-- 当前文件所在文件夹目录 --> <link rel"stylesheet" href"./"> <!-- 当前文件所在父文件夹目录 --> <link rel"stylesheet" href"../">j…...
Sping Cache 使用详解
缓存是提升应用性能的常用手段。它通过将耗时的操作结果存储起来,下次请求可以直接从缓存中获取,从而避免重复计算或查询数据库,显著减少响应时间和服务器负载。Spring 框架提供了强大的缓存抽象 Spring Cache,它简化了缓存的使用…...
动手学深度学习60 机器翻译与数据集
1. 机器翻译与数据集 import os import torch from d2l import torch as d2l#save d2l.DATA_HUB[fra-eng] (d2l.DATA_URL fra-eng.zip,94646ad1522d915e7b0f9296181140edcf86a4f5)#save def read_data_nmt():"""载入“英语-法语”数据集"&qu…...
Python网络爬虫技术
Python网络爬虫技术详解 引言 网络爬虫(Web Crawler),又称网络蜘蛛(Web Spider)或网络机器人(Web Robot),是一种按照一定规则自动抓取互联网信息的程序或脚本。它们通过遍历网页链…...
黑马程序员-redis项目实践笔记1
目录 一、 基于Session实现登录 发送验证码 验证用户输入验证码 校验登录状态 Redis代替Session登录 发送验证码修改 验证用户输入验证码 登录拦截器的优化 二、 商铺查询缓存 缓存更新策略 数据库和缓存不一致解决方案 缓存更新策略的最佳实践方案 实现商铺缓…...
ES-入门聚合查询
url 请求地址 http://192.168.1.108:9200/shopping/_search {"aggs": { //聚合操作"price_group":{ //名称,随意起名"terms":{ //分组"field": "price" //分组字段}}} } 查询出来的结果是 查询结果中价格的平均值 {&q…...
七维大脑: 探索人类认知的未来之路
七维大脑: 探索人类认知的未来之路 随着科技的不断发展,人们对于大脑的认知也在不断扩展。近年来,科学家们提出了一个名为“七维大脑”的概念,试图通过七个维度来理解人类的认知过程。这个概念的提出,让人们开始思考&…...
spring |Spring Security安全框架 —— 认证流程实现
文章目录 开头简介环境搭建入门使用1、认证1、实体类2、Controller层3、Service层3.1、接口3.2、实现类3.3、实现类:UserDetailsServiceImpl 4、Mapper层3、自定义token认证filter 注意事项小结 开头 Spring Security 官方网址:Spring Security官网 开…...
Django+vue自动化测试平台---正式开源!!!
自动化测试:接口、Web UI 与 App 的全面探索 在此郑重声明:本文内容未经本人同意,不得随意转载。若有违者,必将追究其法律责任。同时,禁止对相关源码进行任何形式的售卖行为,本内容仅供学习使用。 Git 地…...
电子电气架构 --- 智能网联汽车未来是什么样子?
我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明自己,无利益不试图说服别人,是精神上的节…...
docker安装elasticsearch(es)+kibana
目录 docker安装elasticsearch 一.准备工作 1.打开docker目录 2.创建elasticsearch目录 3.打开elasticsearch目录 4.拉取elasticsearch镜像 5.检查镜像 二.挂载目录 1.创建数据挂载目录 2.创建配置挂载目录 3.创建插件挂载目录 4.权限授权 三.编辑配置 1.打开con…...
大厂面试真题-说说redis的雪崩、击穿和穿透
缓存雪崩、击穿、穿透是缓存系统中常见的三种问题,它们都会对系统的性能和稳定性造成严重影响。以下是对这三种问题的详细解释以及相应的解决方案: 一、缓存雪崩 问题解释: 缓存雪崩指的是因为某些原因导致缓存中大量的数据同时失效或过期…...
基于算法竞赛的c++编程(28)结构体的进阶应用
结构体的嵌套与复杂数据组织 在C中,结构体可以嵌套使用,形成更复杂的数据结构。例如,可以通过嵌套结构体描述多层级数据关系: struct Address {string city;string street;int zipCode; };struct Employee {string name;int id;…...
C++实现分布式网络通信框架RPC(3)--rpc调用端
目录 一、前言 二、UserServiceRpc_Stub 三、 CallMethod方法的重写 头文件 实现 四、rpc调用端的调用 实现 五、 google::protobuf::RpcController *controller 头文件 实现 六、总结 一、前言 在前边的文章中,我们已经大致实现了rpc服务端的各项功能代…...
在HarmonyOS ArkTS ArkUI-X 5.0及以上版本中,手势开发全攻略:
在 HarmonyOS 应用开发中,手势交互是连接用户与设备的核心纽带。ArkTS 框架提供了丰富的手势处理能力,既支持点击、长按、拖拽等基础单一手势的精细控制,也能通过多种绑定策略解决父子组件的手势竞争问题。本文将结合官方开发文档,…...
oracle与MySQL数据库之间数据同步的技术要点
Oracle与MySQL数据库之间的数据同步是一个涉及多个技术要点的复杂任务。由于Oracle和MySQL的架构差异,它们的数据同步要求既要保持数据的准确性和一致性,又要处理好性能问题。以下是一些主要的技术要点: 数据结构差异 数据类型差异ÿ…...
解决本地部署 SmolVLM2 大语言模型运行 flash-attn 报错
出现的问题 安装 flash-attn 会一直卡在 build 那一步或者运行报错 解决办法 是因为你安装的 flash-attn 版本没有对应上,所以报错,到 https://github.com/Dao-AILab/flash-attention/releases 下载对应版本,cu、torch、cp 的版本一定要对…...
Java面试专项一-准备篇
一、企业简历筛选规则 一般企业的简历筛选流程:首先由HR先筛选一部分简历后,在将简历给到对应的项目负责人后再进行下一步的操作。 HR如何筛选简历 例如:Boss直聘(招聘方平台) 直接按照条件进行筛选 例如:…...
Rapidio门铃消息FIFO溢出机制
关于RapidIO门铃消息FIFO的溢出机制及其与中断抖动的关系,以下是深入解析: 门铃FIFO溢出的本质 在RapidIO系统中,门铃消息FIFO是硬件控制器内部的缓冲区,用于临时存储接收到的门铃消息(Doorbell Message)。…...
dify打造数据可视化图表
一、概述 在日常工作和学习中,我们经常需要和数据打交道。无论是分析报告、项目展示,还是简单的数据洞察,一个清晰直观的图表,往往能胜过千言万语。 一款能让数据可视化变得超级简单的 MCP Server,由蚂蚁集团 AntV 团队…...
Web 架构之 CDN 加速原理与落地实践
文章目录 一、思维导图二、正文内容(一)CDN 基础概念1. 定义2. 组成部分 (二)CDN 加速原理1. 请求路由2. 内容缓存3. 内容更新 (三)CDN 落地实践1. 选择 CDN 服务商2. 配置 CDN3. 集成到 Web 架构 …...
代码随想录刷题day30
1、零钱兑换II 给你一个整数数组 coins 表示不同面额的硬币,另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额,返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位带…...
