文章目录

  • 一、简介
  • 二、使用
    • 1、服务提供者(或者第三方公共):定义接口
    • 2、服务提供者:定义实现类
    • 3、服务提供者:注册服务
    • 4、构建服务提供者jar包
    • 5、客户端:使用 ServiceLoader 来加载服务
  • 三、源码分析
    • 1、源码
    • 2、缺点
  • 四、auto-service库
    • 1、简介
    • 2、有变化的点
  • 参考资料

一、简介

SPI 即 Service Provider Interface ,字面意思就是:“服务提供者的接口”,是JDK内置的一种服务提供发现机制,可以用来启用框架扩展和替换组件,主要是被框架的开发人员使用,比如java.sql.Driver接口,其他不同厂商可以针对同一接口做出不同的实现,MySQL和PostgreSQL都有不同的实现提供给用户,而Java的SPI机制可以为某个接口寻找服务实现。Java中SPI机制主要思想是将装配的控制权移到程序之外,在模块化设计中这个机制尤其重要,其核心思想就是解耦

SPI 将服务接口和具体的服务实现分离开来,将服务调用方和服务实现者解耦,能够提升程序的扩展性、可维护性。修改或者替换服务实现并不需要修改调用方。
在这里插入图片描述

二、使用

1、服务提供者(或者第三方公共):定义接口

public interface HelloSPI {void sayHello();
}

2、服务提供者:定义实现类

public class ImageHello implements HelloSPI {public void sayHello() {System.out.println("Image Hello");}
}public class TextHello implements HelloSPI {public void sayHello() {System.out.println("Text Hello");}
}

3、服务提供者:注册服务

META-INF/services/目录里创建一个以org.example.spi.HelloSPI的文件,这个文件里的内容就是这个接口的具体的实现类。

org.example.spi.ImageHello
org.example.spi.TextHello

4、构建服务提供者jar包

将服务提供者构建为jar包,客户端需要引入

5、客户端:使用 ServiceLoader 来加载服务

import org.example.spi.HelloSPI;import java.util.ServiceLoader;
public class SPIDemo {public static void main(String[] args) {ServiceLoader<HelloSPI> serviceLoader = ServiceLoader.load(HelloSPI.class);// 执行不同厂商的业务实现,具体根据业务需求配置for (HelloSPI helloSPI : serviceLoader) {helloSPI.sayHello();}}
}

我们可以看到,服务正常加载了。

三、源码分析

1、源码

// ServiceLoader实现了Iterable接口,可以遍历所有的服务实现者
public final class ServiceLoader<S> implements Iterable<S>
{// 查找配置文件的目录private static final String PREFIX = "META-INF/services/";// 表示要被加载的服务的类或接口private final Class<S> service;// 这个ClassLoader用来定位,加载,实例化服务提供者private final ClassLoader loader;// 访问控制上下文private final AccessControlContext acc;// 缓存已经被实例化的服务提供者,按照实例化的顺序存储private LinkedHashMap<String,S> providers = new LinkedHashMap<>();// 迭代器private LazyIterator lookupIterator; 
}
// 服务提供者查找的迭代器
public Iterator<S> iterator() {return new Iterator<S>() {Iterator<Map.Entry<String,S>> knownProviders= providers.entrySet().iterator();// hasNext方法public boolean hasNext() {if (knownProviders.hasNext())return true;return lookupIterator.hasNext();}// next方法public S next() {if (knownProviders.hasNext())return knownProviders.next().getValue();return lookupIterator.next();}};
}
// 服务提供者查找的迭代器
private class LazyIterator implements Iterator<S> {// 服务提供者接口Class<S> service;// 类加载器ClassLoader loader;// 保存实现类的urlEnumeration<URL> configs = null;// 保存实现类的全名Iterator<String> pending = null;// 迭代器中下一个实现类的全名String nextName = null;public boolean hasNext() {if (nextName != null) {return true;}if (configs == null) {try {String fullName = PREFIX + service.getName();if (loader == null)configs = ClassLoader.getSystemResources(fullName);elseconfigs = loader.getResources(fullName);} catch (IOException x) {fail(service, "Error locating configuration files", x);}}while ((pending == null) || !pending.hasNext()) {if (!configs.hasMoreElements()) {return false;}pending = parse(service, configs.nextElement());}nextName = pending.next();return true;}public S next() {if (!hasNext()) {throw new NoSuchElementException();}String cn = nextName;nextName = null;Class<?> c = null;try {c = Class.forName(cn, false, loader);} catch (ClassNotFoundException x) {fail(service,"Provider " + cn + " not found");}if (!service.isAssignableFrom(c)) {fail(service, "Provider " + cn  + " not a subtype");}try {S p = service.cast(c.newInstance());providers.put(cn, p);return p;} catch (Throwable x) {fail(service, "Provider " + cn + " could not be instantiated: " + x, x);}throw new Error();          // This cannot happen}
}

首先,ServiceLoader实现了Iterable接口,所以它有迭代器的属性,这里主要都是实现了迭代器的hasNext和next方法。这里主要都是调用的lookupIterator的相应hasNext和next方法,lookupIterator是懒加载迭代器。

其次,LazyIterator中的hasNext方法,静态变量PREFIX就是”META-INF/services/”目录,这也就是为什么需要在classpath下的META-INF/services/目录里创建一个以服务接口命名的文件。

最后,通过反射方法Class.forName()加载类对象,并用newInstance方法将类实例化,并把实例化后的类缓存到providers对象中,(LinkedHashMap<String,S>类型) 然后返回实例对象。

2、缺点

1.不能按需加载,需要遍历所有的实现,并实例化,然后在循环中才能找到我们需要的实现。如果不想用某些实现类,或者某些类实例化很耗时,它也被载入并实例化了,这就造成了浪费。
2.获取某个实现类的方式不够灵活,只能通过 Iterator 形式获取,不能根据某个参数来获取对应的实现类。
3.多个并发多线程使用 ServiceLoader 类的实例是不安全的。
4.没有使用缓存,每次调用load方法都需要重新加载

四、auto-service库

1、简介

auto-service 是 Google 提供的一个 Java 注解处理器库,主要用于简化 Java 服务提供者接口(Service Provider Interface, SPI)的实现。它通过注解自动生成 SPI 配置文件,避免了手动编写 META-INF/services 目录下配置文件的繁琐工作。

它可以:
自动生成 SPI 配置文件
编译时检查 SPI 实现的有效性
简化模块化项目中的服务注册

使用需要先引入依赖包(服务提供者):

        <dependency><groupId>com.google.auto.service</groupId><artifactId>auto-service</artifactId><version>1.1.1</version></dependency>

2、有变化的点

只需要在需要暴露的实现类中,添加注解即可。
并不需要添加 META-INF/services 目录下配置文件了!

import com.google.auto.service.AutoService;// 如果多个接口,value是可以添加数组的
@AutoService(HelloSPI.class)
public class ImageHello implements HelloSPI {public void sayHello() {System.out.println("Image Hello");}
}

我们发现,打出来的服务提供者的jar包,是自带配置文件的:
在这里插入图片描述
这里用的就是Java 的 APT(Annotation Processing Tool)机制

参考资料

https://zhuanlan.zhihu.com/p/84337883

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/917623.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/917623.shtml
英文地址,请注明出处:http://en.pswp.cn/news/917623.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PPT自动化 python-pptx - 10 : 表格(tables)

在日常工作中&#xff0c;我们经常需要制作包含表格的 PowerPoint 演示文稿&#xff0c;以此清晰展示数据或文本信息。手动制作不仅耗时&#xff0c;当数据更新时还需重复操作&#xff0c;效率低下。而 python-pptx 库为我们提供了自动化操作 PowerPoint 表格的可能。本文将详细…

在安卓中使用 FFmpegKit 剪切视频并添加文字水印

在安卓中用到的三方库&#xff1a;https://github.com/arthenica/ffmpeg-kit 这个库很强大&#xff0c;支持很多平台&#xff0c;每个平台都有各自的分支代码&#xff0c;用了一段时间&#xff0c;稳定性挺好的&#xff0c; 找到安卓下的分支&#xff1a;FFmpegKit for Andro…

Flask + HTML 项目开发思路

Flask HTML 项目开发思路&#xff1a;以公共资源交易信息展示为例 一、开篇明义——为什么选 Flask 框架 在众多 Python Web 框架&#xff08;如 Django、Tornado 等&#xff09;里&#xff0c;本次项目坚定选择 Flask&#xff0c;背后有清晰的技术考量&#xff1a; 1. 轻量…

Vue中:deep()和 ::v-deep选择器的区别

在 Vue.js 中&#xff0c;:deep()和 ::v-deep都是用于穿透组件作用域的深度选择器&#xff0c;但它们在语法、适用场景和版本支持上存在区别。以下是两者的核心差异&#xff1a;一、​​语法与用法​ &#xff1a;Vue2中用 ::v-deep&#xff0c;Vue2中不支持:deep()&#xff0c…

Deep learning based descriptor

1、DH3D: Deep Hierarchical 3D Descriptors for Robust Large-Scale 6DoF Relocalization 论文链接 代码链接 这是一篇训练点云的文章&#xff0c;在训练出local descriptor之后&#xff0c;通过聚类的方法得出global descriptor&#xff0c;并且提出了hierarchical network&…

PandasAI连接LLM对MySQL数据库进行数据分析

1. 引言 在之前的文章《PandasAI连接LLM进行智能数据分析》中实现了使用PandasAI连接与DeepSeek模型通过自然语言进行数据分析。不过那个例子中使用的是PandasAI 2.X&#xff0c;并且使用的是本地.csv文件来作为数据。在实际应用的系统中&#xff0c;使用.csv作为库表的情况比…

FloodFill算法——DFS

FloodFill算法就是用来寻找性质相同的连通快的算法&#xff0c;这篇博客都是用dfs来实现FloodFill算法 1.图像渲染 题目链接&#xff1a;733. 图像渲染 - 力扣&#xff08;LeetCode&#xff09; 题目解析&#xff1a;将和&#xff08;sr,sc&#xff09;相连的所有像素相同的…

【BUUCTF系列】[极客大挑战 2019]LoveSQL 1

本文仅用于技术研究&#xff0c;禁止用于非法用途。 Author:枷锁 文章目录一、题目核心漏洞分析二、关键解题步骤与技术解析1. 确定列数&#xff08;ORDER BY&#xff09;2. 联合查询获取表名3. 爆破字段名4. 提取Flag三、漏洞根源与防御方案1. 漏洞成因2. 防御措施四、CTF技巧…

AI时代,童装销售的“指路明灯”

别看现在AI、大数据这些词眼花缭乱的&#xff0c;当年我刚入行那会儿&#xff0c;也跟你一样&#xff0c;对着一堆库存和销量数据发愁&#xff0c;不知道劲儿该往哪使。童装销售这行&#xff0c;看着简单&#xff0c;其实水挺深。不过呢&#xff0c;这二十多年摸爬滚打下来&…

Swin-Transformer从浅入深详解

第一部分&#xff1a;出现背景在 Swin Transformer 出现之前&#xff0c;计算机视觉&#xff08;Computer Vision, CV&#xff09;领域主要由 CNN (卷积神经网络) 主导。后来&#xff0c;NLP&#xff08;自然语言处理&#xff09;领域的 Transformer 模型被引入 CV&#xff0c;…

如何手动打包 Linux(麒麟系统)的 Qt 程序

gcc版本 gcc版本确保目标系统&#xff08;运行环境&#xff09;的 GCC 版本 高于或等于开发环境的版本&#xff0c;否则程序无法在目标平台运行。通过 gcc -v 可查看当前版本。cmake生成可执行文件 强烈建议在cmakelists添加设置运行时 rpath 为 $ORIGIN/…/lib&#xff08;相对…

解决 “crypto.hash is not a function”:Vite 从 6.x 升级至 7.x 后 `pnpm run dev` 报错问题

&#x1f680; 作者主页&#xff1a; 有来技术 &#x1f525; 开源项目&#xff1a; youlai-mall ︱vue3-element-admin︱youlai-boot︱vue-uniapp-template &#x1f33a; 仓库主页&#xff1a; GitCode︱ Gitee ︱ Github &#x1f496; 欢迎点赞 &#x1f44d; 收藏 ⭐评论 …

我的创作纪念日____在 CSDN一年来的成长历程和收获

365 天创作札记&#xff1a;在代码与文字的褶皱里&#xff0c;遇见 1300 束光一年来。点开csdn网站后台粉丝数的那一刻&#xff0c;1327 这个数字在屏幕上微微发烫。原来那些在深夜敲下的字符、调试到凌晨的代码示例、反复修改的技术拆解&#xff0c;真的在时光里悄悄织成了一张…

VirtualBox 的 HOST 键(主机键)是 右Ctrl 键(即键盘右侧的 Ctrl 键)笔记250802

VirtualBox 的 HOST 键&#xff08;主机键&#xff09;是 右Ctrl 键&#xff08;即键盘右侧的 Ctrl 键&#xff09;笔记250802 VirtualBox 的 HOST 键&#xff08;主机键&#xff09;是什么?HOST键 是 右Ctrl 键VirtualBox 的 主机键&#xff08;Host Key&#xff09; 是一个…

Zama的使命

全同态加密&#xff08;Fully Homomorphic Encryption&#xff0c;FHE&#xff09;实现互联网端到端加密的使命的重要里程碑。(FHE) 是一种无需解密即可处理数据的技术。它可用于在公共、无需许可的区块链上创建私人智能合约&#xff0c;只有特定用户才能看到交易数据和合约状态…

Go语言流式输出技术实现-服务器推送事件(Server-Sent Events, SSE)

目录引言背景与技术概述实现技术细节1. HTTP 头部配置2. 事件格式与发送3. 保持连接与刷新4. 处理连接关闭4.1 使用上下文管理连接生命周期4.2 使用通道管理客户端连接5. 客户端交互6.demo7.Go转发大模型流式输出demo引言 服务器推送事件&#xff08;Server-Sent Events, SSE&…

高端房产管理小程序

系统介绍1、用户端地图找房&#xff1a;对接地图API&#xff0c;地图形式显示周边房源,支持新盘和租房两种模式查询房价走势&#xff1a;城市房价走势&#xff0c;由后台每月录入房源搜索&#xff1a;搜索房源&#xff0c;支持多维度筛选房源类型&#xff1a;新盘销售、房屋租赁…

文本转语音(TTS)脚本

文本转语音(TTS)脚本 概述 generate_voice.py 是一个用于生成语音的Python脚本。该脚本提供了文本转语音(TTS)功能&#xff0c;可以将文本内容转换为语音文件。 功能特性 文本转语音: 将输入的文本转换为语音文件多种语音选项: 支持不同的语音类型和参数批量处理: 可以处理多个…

磁盘管理与分区

磁盘管理 一、磁盘类型 SATA,SCSI,SAS类型的磁盘&#xff0c;在Linux中用sd来表示。 其中第一块硬盘为sda&#xff0c;第二块二sdb&#xff0c;以此类推。 第一块硬盘的第一个分区为sda1。 nvme类型的磁盘&#xff0c;在Linux中使用nvmeXnYpZ进行表示。 X&#xff1a;数字&…

Linux 逻辑卷管理

练习创建物理卷(pv->vg->lv)物理卷&#xff08;PV&#xff09;就像把一块块独立的硬盘&#xff0c;标记成 "可用于搭建 LVM 的积木"&#xff0c;让系统知道这些硬盘可以被 LVM 管理。#把sdb这块硬盘标记为物理卷&#xff08;相当于给这块积木盖章&#xff0c;说…