0

0

在Java里什么是SPI_Java服务提供者接口解析

P粉602998670

P粉602998670

发布时间:2026-02-07 15:32:03

|

170人浏览过

|

来源于php中文网

原创

java.util.ServiceLoader 是 Java SPI 机制的核心实现,用于运行时按 META-INF/services/ 下配置加载接口实现类。它基于约定路径查找、解析类名、反射实例化,依赖类加载器与正确资源位置,不提供依赖注入或插件管理功能。

在java里什么是spi_java服务提供者接口解析

什么是 java.util.ServiceLoader

ServiceLoader 是 Java 标准库中实现 SPI(Service Provider Interface)机制的核心类,它不负责定义接口,也不提供实现,只做一件事:在运行时按约定路径查找、加载并实例化 META-INF/services/ 下声明的接口实现类。

你写一个接口(比如 ImageProcessor),让不同模块提供各自的实现(JpegProcessorPngProcessor),再在各自 JAR 包里放一个 META-INF/services/com.example.ImageProcessor 文件,内容就是实现类全限定名。然后用 ServiceLoader.load(ImageProcessor.class) 就能拿到所有可用实现。

关键点:SPI 不是“插件系统”,也不是“依赖注入”,它本质是 JDK 提供的一种弱耦合的、基于文件配置的发现机制。

为什么 ServiceLoader 加载不到我的实现类?

常见原因不是代码写错了,而是资源路径或类加载器没对上:

立即学习Java免费学习笔记(深入)”;

闪电说
闪电说

AI语音输入法

下载
  • META-INF/services/ 必须在 classpath 根路径下(即和 com/ 包同级),不是放在 src/main/resources 就自动生效——要确认编译后该文件真出现在 JAR 或 classes 输出目录的根下
  • 服务配置文件名必须是接口的**完整类名**(如 com.example.LogHandler),不能有空格、大小写错误,也不能带 .class
  • 文件内容只能是实现类的全限定名,每行一个,末尾不能有空格或 BOM 字符(Windows 记事本容易加 BOM)
  • 如果用模块系统(module-info.java),需显式声明 uses com.example.ImageProcessor;(调用方)和 provides com.example.ImageProcessor with com.example.JpegProcessor;(提供方)
  • ServiceLoader 默认使用当前线程上下文类加载器(Thread.currentThread().getContextClassLoader()),若在 Web 容器或 OSGi 环境中,这个加载器可能看不到你的服务文件

ServiceLoaderClassLoader.getSystemResources() 有什么区别

两者都能查资源,但语义和用途完全不同:

  • ServiceLoader 是高层封装:它读取 META-INF/services/ 文件 → 解析类名 → 用指定类加载器 Class.forName(...) → 调用无参构造器实例化 → 还支持 reload()iterator() 惰性加载
  • ClassLoader.getSystemResources("META-INF/services/com.example.Foo") 只返回 URL 列表,你需要自己读文件、解析、反射、处理异常,且无法保证类加载器一致性
  • 性能上,ServiceLoader 第一次 load() 会缓存结果;而手动遍历资源每次都要 IO + 解析,还容易漏掉多模块同名服务的合并逻辑
  • 兼容性上,ServiceLoader 自 Java 6 引入,Java 9+ 在模块化下仍保持行为一致(只要模块声明了 uses/provides);手写资源查找在模块路径下很可能失效

Spring 的 @SPI 或 Dubbo 的 SPI 是一回事吗?

不是。Java 原生 SPI 很简单,只有加载和实例化;而 Spring、Dubbo、Apache ShardingSphere 等框架的 “SPI” 是在其基础上做的增强,通常包含:

  • 支持带参数的构造器、工厂方法、依赖注入(原生 SPI 只支持无参构造)
  • 支持扩展点别名(如 dubbo:// 对应 DubboProtocol)、默认实现、条件加载(@ConditionalOnClass
  • 缓存策略更细(按 key 缓存、热加载)、异常兜底(找不到实现时返回空或默认)、优先级排序
  • 配置方式不局限于 META-INF/services,可能读 spring.factories 或自定义配置文件

所以看到框架文档说 “基于 SPI 实现”,实际用的大概率是自家重写的加载器,只是借用了“服务发现”这个概念。直接套用原生 ServiceLoader 逻辑去调试 Dubbo 扩展点,基本会失败。

真正容易被忽略的是:SPI 的“服务”不是功能模块,而是契约。接口设计是否稳定、版本如何兼容、实现类是否线程安全、是否允许重复注册——这些都得靠人来约定,ServiceLoader 本身不校验、不约束、也不报错。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

118

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

72

2026.01.26

dubbo和zookeeper有什么区别
dubbo和zookeeper有什么区别

dubbo和zookeeper的区别:1、功能定位;2、使用场景;3、数据存储与协调;4、集成与关系;5、性能与可靠性;6、扩展性与灵活性;7、社区与生态系统。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

227

2024.02.23

springcloud和dubbo有哪些区别
springcloud和dubbo有哪些区别

springcloud和dubbo的区别:1、定位与关注点;2、生态环境与集成性;3、调用方式与性能;4、组件与功能;5、定制性与灵活性;6、学习曲线与上手难度;7、社区支持与维护。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

126

2024.02.23

dubbo原理和机制是什么
dubbo原理和机制是什么

dubbo原理和机制的解释:1、核心组件;2、通信原理;3、集群容错;4、自动发现与注册;5、负载均衡与路由;6、序列化与传输;7、监控与日志;8、扩展性;9、安全性;10、与spring集成等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

102

2024.02.23

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1295

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

276

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2200

2025.12.29

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.3万人学习

C# 教程
C# 教程

共94课时 | 8.8万人学习

Java 教程
Java 教程

共578课时 | 59.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号