Java热插拔技术之SPI

news2025/1/11 4:15:25

文章目录

  • 背景
  • SPI是什么
  • SPI和API的区别
  • Java SPI
  • 实践出真知
  • 总结

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。

背景

最近,公司需要针对一个使用C#的系统以插件的形式进行二次开发。系统提供了一个类库,我们只需要实现类库中的接口,并实现相应的方法,即可完成一个插件的开发。

然后,我们将实现类打包成dll文件,这个dll文件就像是Java中的jar包一样。我们将这个dll文件上传到指定的文件夹,系统会热更新并加载这个dll文件。之后,我们需要在系统中配置该类的全限定名,这样插件就可以生效了。

这种插件式开发方式非常有趣,我在想Java能否实现这样的动态扩展。经过一番查询和研究,我发现Java确实提供了类似的功能,那就是SPI,本文将对SPI进行一个简单的介绍。

SPI是什么

SPI全称Service Provider Interface,是一种服务提供者接口,定义了一组用于实现特定服务的接口或抽象类。它是一种动态替换发现的机制,使得接口和实现可以分离。SPI被专门提供给服务提供者或扩展框架功能的开发者去使用,由调用方提供接口,第三方(提供方)来实现。

SPI和API的区别

SPI(Service Provider Interface)API(Application Programming Interface) 有以下几个区别:

  1. 定义方式:API可以是一组函数、类、协议或工具,通常易于理解和使用。API 定义了可供开发者调用的公共方法和功能。而 SPI 是一种机制,用于揭示和加载服务提供者的实现,它通常以接口的形式存在。

  2. 使用方式:API 是开发者在应用程序中直接调用的,开发者通过使用 API 提供的方法和功能来实现特定的业务逻辑。而 SPI 是通过类加载器和反射机制动态加载和实例化服务提供者的,开发者无需直接调用 SPI 的机制,而是通过使用 SPI 加载的服务实例来实现特定的功能。

  3. 扩展性:API 的设计目的是为了提供一套公共的接口和方法,以便开发者进行二次开发和扩展。而 SPI 的目的是为了实现解耦合和动态加载,允许第三方服务提供者通过 SPI 机制向应用程序中注入其实现,从而实现功能的扩展和替换。

总的来说,API是面向使用者的,它定义了如何使用软件组件;而SPI是面向提供者的,它定义了如何为软件组件提供服务 。

Java SPI

Java SPI是Java 6引入的一种服务发现机制。主要包括以下4个核心概念:

  1. 服务接口:定义一组对外提供服务的服务接口,通常以接口或抽象类的形式存在。
  2. 服务提供者接口:服务接口的具体实现类,提供给应用程序使用。
  3. 配置文件META-INF/services 目录下的配置文件,用于声明服务提供者的实现类。文件名称为服务接口的全限定名,文件内容为服务提供者接口的全限定名。
  4. ServiceLoader :Java SPI关键类,用于加载服务提供者接口的服务。 ServiceLoader 中有各种实用方法,用于获取特定的实现、迭代它们或再次重新加载服务。

当应用程序需要使用某项服务时,它会在类路径下查找 META-INF/services 目录下的配置文件,文件中列出了对应服务接口的实现类。然后,应用程序可以通过 Java 标准库提供的 ServiceLoader 类动态加载并实例化这些服务提供者,从而使用其功能。这种机制被广泛应用于 Java 中各种框架和组件的扩展开发,例如数据库驱动、日志模块等。

Java SPI的优缺点:

优点:

  1. 实现解耦:SPI使得接口的定义与具体业务实现分离,使得应用程序可以根据实际业务情况启用或替换具体组件。
  2. 可插拔性和可扩展性:SPI 允许第三方提供新的服务实现模块,并通过配置文件进行声明,在运行时动态加载,这样可以轻松地扩展和替换系统中的功能模块,实现了可插拔性和可扩展性。

缺点:

  1. 无法按需加载:虽然ServiceLoader做了延迟载入,但是基本只能通过遍历全部获取,也就是接口的实现类得全部载入并实例化一遍。如果你并不想用某些实现类,或者某些类实例化很耗时,它也被载入并实例化了,这就造成了浪费。
  2. 获取某个实现类的方式不够灵活:只能通过Iterator形式获取,不能根据某个参数来获取对应的实现类。
  3. 不提供类似于Spring的IOC和AOP功能:扩展如果依赖其他的扩展,做不到自动注入和装配。
  4. 并发问题:多个并发多线程使用ServiceLoader 类的实例是不安全的。加载不到实现类时抛出并不是真正原因的异常,错误很难定位。

Java 提供了许多 SPI,以下是SPI及其提供的服务的一些示例:

  • java.util.spi.CurrencyNameProvider:为Currency类提供本地化的货币符号。
  • java.util.spi.LocaleNameProvider: 为 Locale 类提供本地化名称。
  • java.sql.Driver:从 4.0 版本开始,JDBC API 支持 SPI 模式。旧版本使用 Class.forName() 方法加载驱动程序。
  • jakarta.persistence.spi.PersistenceProvider:提供JPA API的实现。
  • java.text.spi.DateFormatProvider:为指定区域设置的日期和时间格式。
  • java.util.spi.TimeZoneNameProvider:为 TimeZone 类提供本地化时区名称。
  • javax.sound.sampled.spi:Java声音SPI接口,开发者可以提供自定义的音频文件读取和写入插件。
  • javax.imageio.spi:Java图像I/O的SPI接口,开发者可以提供自定义的图像读取和写入插件。

实践出真知

下面我们用一个搜索电影的例子来写一个Java SPI的示例。有时候一个人心血来潮想看一些武打电影,我们可以通过UC搜索,UC资源有限,有时就需要使用魔法通过谷歌搜索来找电影,反正不管是黑猫白猫,抓到老鼠就是好猫。所以,可以提供一个搜索电影的服务接口,具体使用哪个搜索就看个人喜好了。

  1. 定义服务接口:首先,新建一个spi-provider 模块,定义一个电影提供者接口MovieProvider,里面有一个方法searchMovie()
public interface MovieProvider {

    /**
     * 搜索电影
     * @param movieName 电影名
     */
    void searchMovie(String movieName);
}
  1. 实现服务接口:新建两个模块 uc-providergoogle-provider,分别引入spi-provider 模块,并在各自模块实现MovieProvider接口。

引入spi-provider 模块:

 <dependency>
     <groupId>BasicJava</groupId>
     <artifactId>spi-provider</artifactId>
     <version>1.0-SNAPSHOT</version>
 </dependency>
public class UCProvider implements MovieProvider{
    /**
     * 搜索电影
     *
     * @param movieName 电影名
     */
    @Override
    public void searchMovie(String movieName) {
        System.out.println("通过UC搜索:"+movieName);
    }
}
public class GoogleProvider implements MovieProvider{
    /**
     * 搜索电影
     *
     * @param movieName 电影名
     */
    @Override
    public void searchMovie(String movieName) {
        System.out.println("通过谷歌搜索:"+movieName);
    }
}
  1. 配置服务提供者:新建一个 spi-consumer 模块,引入 uc-providergoogle-provider, 并在 META-INF/services 目录下,创建一个以MovieProvider接口全限定名为命名的文本文件。
<dependencies>
    <dependency>
        <groupId>BasicJava</groupId>
        <artifactId>uc-provider</artifactId>
        <version>1.0-SNAPSHOT</version>
    </dependency>

    <dependency>
        <groupId>BasicJava</groupId>
        <artifactId>google-provider</artifactId>
        <version>1.0-SNAPSHOT</version>
    </dependency>
</dependencies>

META-INF/services
site.sunlong.provider.MovieProvider

  1. 使用服务:一切准备完毕,接下来就是使用它们的时候了。怎么使用呢?为了实现动态使用插件的效果,在这里我们以配置文件的形式使用它,在配置文件中配置UCProviderGoogleProvider是否生效,然后动态的获取配置文件,话不多说,请看实操。

创建 spi-config.properties配置文件类:
spi-config.properties
spi-config.properties配置文件内容如下:

site.sunlong.provider.GoogleProvider=enable
site.sunlong.provider.UCProvider=enable

spi-consumer 模块新建一个SpiConsumer用于测试,在该类中每10s读取一次配置文件的内容,判断插件是否生效,最后只调用生效的插件。具体代码如下:

public class SpiConsumer {
    private static ScheduledExecutorService executor = Executors.newScheduledThreadPool(1);
    public static void main(String[] args) {

        ServiceLoader<MovieProvider> providers = ServiceLoader.load(MovieProvider.class);
        String movieName = "波多";

        executor.scheduleAtFixedRate(() -> {
            // 读取配置文件
            Properties properties = new Properties();
            ClassLoader classLoader = SpiConsumer.class.getClassLoader();
            try (InputStream input = classLoader.getResourceAsStream("spi-config.properties")) {
                if (input == null) {
                    System.out.println("无法找到配置文件");
                    return;
                }
                properties.load(input);
            } catch (IOException e) {
                e.printStackTrace();
                return;
            }

            for (MovieProvider provider : providers) {
                String name = provider.getClass().getName();
                String property = properties.getProperty(name);
                if ("enable".equals(property)) {
                    provider.searchMovie(movieName);
                }
            }
            System.out.println(DateTime.now());
            System.out.println();
        }, 0, 10, TimeUnit.SECONDS); // 初始延迟为0,之后每隔10秒执行一次
    }
}

先后将site.sunlong.provider.GoogleProvidersite.sunlong.provider.UCProvider设置为disable,输出结果如下:

通过谷歌搜索:波多
通过UC搜索:波多
2023-11-21T23:08:58.241+08:00

通过谷歌搜索:波多
通过UC搜索:波多
2023-11-21T23:09:08.213+08:00

通过UC搜索:波多
2023-11-21T23:09:18.200+08:00

通过UC搜索:波多
2023-11-21T23:09:28.215+08:00

通过UC搜索:波多
2023-11-21T23:09:38.208+08:00

通过谷歌搜索:波多
2023-11-21T23:09:48.202+08:00

通过谷歌搜索:波多
2023-11-21T23:09:58.210+08:00

打印结果虽然有延迟,但从结果中还是可以看出我们是实现了可插拔插件的功能,只是结果有瑕疵,可以把配置文件放到缓存或者数据库中以便准确的判断插件是否生效。

总结

总的来说,Java SPI 的实现原理是通过类加载器动态加载配置文件,再利用反射机制实例化具体的服务提供者,并将其注入到应用程序中供服务消费者使用。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

中仕公考:2024年度河南省公务员考试公告发布!共招录9900人!

河南省2024年度统一考试录用公务员公告于今日发布&#xff0c;共计划招录9900人。 报名时间&#xff1a;1月18日9&#xff1a;00-1月24日17&#xff1a;00 笔试时间&#xff1a;3月16日-3月17日 报名方式&#xff1a;登录“河南人事考试网”进行网上报名 2024年省考29.5%的…

CSS中的width与height

CSS中的width与height 1 display: inline-block2 width: auto2.1 外部尺寸与流体特性2.1.1 正常流宽度2.1.2 格式化宽度 2.2 内部尺寸与流体特性2.2.1 包裹性2.2.2 首选最小宽度2.2.3 最大宽度 3 height: 100%3.1 如何让元素支持height: 100%效果 1 display: inline-block 我们…

健康之钥:新生儿维生素K的呵护指南

引言&#xff1a; 维生素K&#xff0c;在新生儿的成长旅程中扮演着不可忽视的角色。它对于血液凝结和骨骼发育至关重要。本文将深入探讨维生素K的功能、补充时机&#xff0c;以及在给新生儿补充维生素K时应该注意的事项&#xff0c;为小天使们提供最贴心的呵护。 第一部分&…

动态内存面试的经典题目

&#x1d649;&#x1d65e;&#x1d658;&#x1d65a;!!&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦ &#x1f44f;&#x1f3fb;‧✧̣̥̇:Solitary-walk ⸝⋆ ━━━┓ - 个性标签 - &#xff1a;来于“云”的“羽球人”。…

大创项目推荐 深度学习疲劳驾驶检测 opencv python

文章目录 0 前言1 课题背景2 实现目标3 当前市面上疲劳驾驶检测的方法4 相关数据集5 基于头部姿态的驾驶疲劳检测5.1 如何确定疲劳状态5.2 算法步骤5.3 打瞌睡判断 6 基于CNN与SVM的疲劳检测方法6.1 网络结构6.2 疲劳图像分类训练6.3 训练结果 7 最后 0 前言 &#x1f525; 优…

Resize:最近邻插值、双线性插值、双三次插值

Resize&#xff1a;最近邻插值、双线性插值、双三次插值 Opencv resize函数1. 最近邻插值&#xff08;INTER_NEAREST&#xff09;1.1 原理1.2 代码实例1.3 简单的代码复现1.4 特点 2. 双线性插值&#xff08;INTER_LINEAR&#xff09;&#xff08;默认值&#xff09;2.1 原理2.…

Rust-解引用

“解引用”(Deref)是“取引用”(Ref)的反操作。取引用&#xff0c;我们有&、&mut等操作符&#xff0c;对应的&#xff0c;解引用&#xff0c;我们有操作符&#xff0c;跟C语言是一样的。示例如下&#xff1a; 比如说&#xff0c;我们有引用类型p:&i32;,那么可以用符…

[笔记]深度学习入门 基于Python的理论与实现(三)

代码仓库 gitee 3. 神经网络 神经网络的出现就是为了解决设定权重的工作&#xff0c;即机器自动从数据中学习&#xff0c;确定合适的、能符合预期的输入与输出的权重。 3.1 从感知机到神经网络 神经网络和感知机有很多共同点&#xff0c;这里主要介绍差异 3.1.1 神经网络例子…

数据库|数据库范式(待完成)

文章目录 数据库的范式数据库的基本操作什么是数据库的范式产生的背景&#xff08;没有规范化的坏处/带来的问题&#xff09;规范化表格设计的要求五大范式的作用——树立标准打个比方——桥的承载能力1NF&#xff08;1范式&#xff09;如何转换成合适的一范式 2NF&#xff08;…

Flask 项目怎么配置并创建第一个小项目?附上完成第一个小案例截图

目录 1. 为什么要学习 flask&#xff1f; 2. flask 是什么&#xff1f; 3. flask 如何使用&#xff1f; 要安装 Flask&#xff0c;可以按照以下步骤进行&#xff1a; 4. 使用流程 4.1. 新建项目 4.1.1. 打开 pycharm&#xff0c;新建项目 4.1.2. 设置目录&#xff0c;并…

centos7 arm服务器编译安装PaddlePaddle

前言 随着国产服务器发展&#xff0c;部署项目需要用在国产服务器上&#xff0c;官方教程里面很多没有讲解到&#xff0c;安装过程中出现了各种各样的问题&#xff0c;以下是对官方教程的补充&#xff0c;有什么问题&#xff0c;欢迎指正&#xff01; 一、环境准备 gcc: 8.2版…

Python 网络编程之TCP详细讲解

【一】传输层 【1】概念 传输层是OSI五层模型中的第四层&#xff0c;负责在网络中的两个端系统之间提供数据传输服务主要协议包括**TCP&#xff08;传输控制协议&#xff09;和UDP&#xff08;用户数据报协议&#xff09;** 【2】功能 **端到端通信&#xff1a;**传输层负责…

数学建模-预测人口数据

目录 中国09~18年人口数据 创建时间 绘制时间序列图 使用专家建模器 得到结果 预测结果 残差的白噪声检验 中国09~18年人口数据 创建时间 路径&#xff1a;数据-> 定义日期和时间 绘制时间序列图 使用专家建模器 看看spss最终判断是那个模型最佳的契合 得到结果 预…

什么是NTFS格式文件系统?Tuxera NTFS for Mac2024下载步骤

一般磁盘格式分为&#xff1a;FAT、FAT32、NTFS&#xff0c;这几种格式目前是我们最常遇到的文件系统格式&#xff0c;其中现在遇到最多的就是NTFS格式&#xff0c;为更好地了解这类文件系统格式&#xff0c;小编今天专门介绍一下什么是NTFS格式文件系统以及它的特点和局限性。…

网络安全ctf比赛/学习资源整理,【解题工具、比赛时间、解题思路、实战靶场、学习路线】推荐收藏!

对于想学习或者参加CTF比赛的朋友来说&#xff0c;CTF工具、练习靶场必不可少&#xff0c;今天给大家分享自己收藏的CTF资源&#xff0c;希望能对各位有所帮助。 CTF在线工具 首先给大家推荐我自己常用的3个CTF在线工具网站&#xff0c;内容齐全&#xff0c;收藏备用。 1、C…

Chapter 10 类的继承(上篇)

目的&#xff1a;了解三种继承方式&#xff0c;并清楚其中的差别 &#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;&#x1f383;…

【JVM】垃圾回收 GC

一、前言 垃圾回收&#xff08;Garbage Collection&#xff0c;GC&#xff09;是由 Java 虚拟机&#xff08;JVM&#xff09;垃圾回收器提供的一种对内存回收的一种机制&#xff0c;它一般会在内存空闲或者内存占用过高的时候对那些没有任何引用的对象不定时地进行回收。以避免…

imgaug库指南(26):从入门到精通的【图像增强】之旅(万字长文!)

引言 在深度学习和计算机视觉的世界里&#xff0c;数据是模型训练的基石&#xff0c;其质量与数量直接影响着模型的性能。然而&#xff0c;获取大量高质量的标注数据往往需要耗费大量的时间和资源。正因如此&#xff0c;数据增强技术应运而生&#xff0c;成为了解决这一问题的…

在uni-app中使用sku插件,实现商品详情页规格展示和交互。

商品详情 - SKU 模块 学会使用插件市场&#xff0c;下载并使用 SKU 组件&#xff0c;实现商品详情页规格展示和交互。 存货单位&#xff08;SKU&#xff09; SKU 概念 存货单位&#xff08;Stock Keeping Unit&#xff09;&#xff0c;库存管理的最小可用单元&#xff0c;通…

电子签章服务器,如何解决无纸化最后一公里?

钉钉、飞书、企微、OA、ERP等主流企业办公系统&#xff0c;无法实现电子签章&#xff0c;往往审批后还要将合同文件打印出来再进行签章。实现无纸化办公的这最后一公里就成了难题。电子签章服务器的出现&#xff0c;提供了完美的解决方案。本文将从专业角度&#xff0c;探讨电子…