JAVA中的SPI思想介绍

目录
  • 1. SPI介绍
  • 2. SPI规则
  • 3. SPI案例
    • 3.1 组件的定义
    • 3.2 组件的实现
    • 3.3 组件的选用
  • 4. SPI原理
  • 5. SPI要求
  • 6. SPI应用
  • 总结

1. SPI介绍

SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的接口,其意义在于为某个接口寻找服务的实现,主要应用在框架中用来寻找组件,提高扩展性。

汽车制造是一个比较繁琐的过程,通常的手段是先规定汽车各个零部件的生产规格,各个零部件厂商按照这种规则去生产合格的零部件。汽车生产商挑选合适的零部件去组装以出产汽车。汽车某个零部件损坏也不用废弃掉整个汽车,只需要更换组件即可。

SPI就是用来怎么去寻找汽车零部件的一种机制,生产规格就是接口的定义,零部件生产商生产零部件就是遵循接口提供具体的实现,SPI挑选合适的组件进行组装后完成特定的功能,当某个组件存在漏洞或问题时可以在不改动代码的前提下替换组件以提高扩展性。

2. SPI规则

SPI旨在为某个接口寻找服务的实现,因此在设计初期就要规定好组件的接口,JAVA的SPI机制使用步骤如下:

定义组件接口(生产规格)

实现组件接口(提供组件)

配置组件文件(查找组件)

反射实例调用(组装工作)

3. SPI案例

组件定义工程中定义组件开发的规范,即定义组件需要实现哪些接口组件A工程和组件B工程提供对组件的实现,即实现组件定义的接口组件应用工程挑选合适的组件进行组件的运用

3.1 组件的定义

在【commons-api】工程中定义组件的规范,即定义接口,接口名称为ComponentService,内容如下:

public interface ComponentService
{
    /**
     * 获取组件的名称
     * @return 组件名称
     */
    String getComponentName();
}

3.2 组件的实现

在【component-A】工程中按照组件定义的规范提供组件,即实现组件定义接口,类名称为ComponentA,内容如下:

public class ComponentA implements ComponentService
{
    /**
     * 组件名称
     */
    private static final String COMPONENT_NAME = "组件A";

    @Override
    public String getComponentName()
    {
        return COMPONENT_NAME;
    }
}

按照JAVA的SPI规则在【component-A】工程的resource/META-INF/services资源目录下新建文件,文件名称为组件接口的全限定名,内容为组件实现的全限定名

在【component-B】工程中也提供对应的组件实现,类名称为ComponentB,内容如下:

public class ComponentB implements ComponentService
{
    /**
     * 组件名称
     */
    private static final String COMPONENT_NAME = "组件B";

    @Override
    public String getComponentName()
    {
        return COMPONENT_NAME;
    }
}

同样在【component-B】工程的resource/META-INF/services资源目录下配置文件

3.3 组件的选用

基于maven构建的java工程使用pom文件来编排项目所需要的依赖组件,现在需要用到组件,并且我觉得A组件比B组件更适合我,如是在【component-application】工程的pom中我编排了组件A,内容如下:

<dependencies>
    <dependency>
        <groupId>com.xxxx</groupId>
        <artifactId>component-A</artifactId>
        <version>1.0-SNAPSHOT</version>
    </dependency>
</dependencies>

在【component-application】工程中新建应用程序启动类来运用组件,类名称为Main,内容如下:

public class Main
{
    public static void main(String[] args)
    {
        // 加载组件
        ServiceLoader<ComponentService> components = ServiceLoader.load(ComponentService.class);
        for (ComponentService component : components)
        {
            // 运用组件:打印组件名称
            System.out.println(component.getComponentName());
        }
    }
}

启动【component-application】工程的Main类的main方法,结果如下:

假如某一天我发现组件A存在很大漏洞,需要更换组件将组件A替换成组件B,只需要在【component-application】工程的pom中去掉组件A的依赖,导入组件B的依赖即可,pom内容如下:

<dependencies>
    <dependency>
        <groupId>com.xxxx</groupId>
        <artifactId>component-B</artifactId>
        <version>1.0-SNAPSHOT</version>
    </dependency>
</dependencies>

无需修改【component-application】工程的具体使用细节,就可以达到替换组件的目的,运行如下:

4. SPI原理

JAVA提供的SPI机制主要依靠的是java.util.ServiceLoader类,从SPI案例中入手,进入ServiceLoader.load方法一探究竟。

load方法最终会创建一个LazyIterator 的实例,看到Iterator大概就知道和迭代器有关,继续了解一下LazyIterator 是什么

猜想得不错,LazyIterator 和迭代器有关,它是ServiceLoader的一个内部类,实现了Iterator接口,那只需要重点关注Iterator接口定义的方法

public boolean hasNext()
public S next()

Iterator接口定义的hasNext方法用于判断迭代的是否是否还有下一个元素,LazyIterator 的hasNext方法最终是调用的hasNextService方法,重点研究这个。

通过类加载器获取到指定目录下的资源文件配置的组件实现的全限定名,存放在configs的一个容器变量中,有了组件实现的全限定名,后面多半就是反射生成对象返回了,继续看一下LazyIterator 的next方法。LazyIterator的next方法主要逻辑是在nextService方法中,仔细分析一下

和刚才的猜想一致,拿到了组件实现的全限定名通过Class.forName来生成组件对象,所以程序就通过for循环得到了对象可以进行调用。

5. SPI要求

java.util.ServiceLoader类是这样来描述自己的

A simple service-provider loading facility.

一个简单的服务提供者加载工具

The only requirement enforced by this facility is thatprovider classes must have a zero-argument constructor so that they can beinstantiated during loading

强制执行的唯一要求是提供者类必须有一个无参的构造函数,以便它们可以在加载过程中实例化,从Class.forName生成实例对象就可以看出使用的是无参构造

6. SPI应用

​ SPI的这种组件替换思想很容易让人想到我们熟知的JDBC规范。JDBC是JAVA规定的应用程序连接数据库的标准,定义了连接数据库的几个接口,比如Connection、Statement、ResultSet。各个数据库厂商提供自己的JDBC实现,这就是我们所说的数据库驱动。开发人员只需要关心如何使用JDBC的各个接口,不需要学习不同厂商的实现,这就是面向接口编程。

JDBC编程分为四个步骤,SPI在驱动管理器DriverManager中得到了应用

Mysql驱动和SqlServer驱动都有SPI的配置

在驱动管理器的loadInitialDrivers方法中就会通过SPI机制获取可用的驱动,loadInitialDrivers方法会在静态代码块中被调用。这里并没有通过全限定名反射实例化,真正的驱动注册是数据库厂商提供的驱动类中通过静态代码块将驱动注册到驱动管理器中的registeredDrivers集合变量中的,以MySQL驱动为例:

在驱动管理器的getConnection方法中会遍历SPI查找到的可用驱动,通过驱动获取链接,直至链接获取成功才返回。

总结

到此这篇关于JAVA中的SPI思想介绍的文章就介绍到这了,更多相关JAVA SPI思想内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java SPI机制详细介绍

    目录 为什么需要SPI? 什么是SPI?SPI和API的区别 来人,上点对抗 spi-provider spi-user 总结 为什么需要SPI? 思考一个场景,我们封装了一套服务,别人通过引入我们写好的包,就可以使用这些接口API,完成相应的操作,这本来没有什么问题,但是会存在使用该服务的实体有不相同的业务需求,需要进一步的扩展,但是由于api是写好的,想要扩展并非那么的简单,如果存在这样子的场景,我们该怎么办? 可以使用Java 提供的SPI机制 什么是SPI?SPI和API的区别 SPI

  • 一文搞懂Java的SPI机制(推荐)

    目录 1 简介 缺点 源码 使用 适用场景 插件扩展 案例 1 简介 SPI,Service Provider Interface,一种服务发现机制. 有了SPI,即可实现服务接口与服务实现的解耦: 服务提供者(如 springboot starter)提供出 SPI 接口.身为服务提供者,在你无法形成绝对规范强制时,适度"放权" 比较明智,适当让客户端去自定义实现 客户端(普通的 springboot 项目)即可通过本地注册的形式,将实现类注册到服务端,轻松实现可插拔 缺点 不能按需

  • Java SPI用法案例详解

    1.什么是SPI      SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的接口,它可以用来启用框架扩展和替换组件. SPI的作用就是为这些被扩展的API寻找服务实现. 2.SPI和API的使用场景     API (Application Programming Interface)在大多数情况下,都是实现方制定接口并完成对接口的实现,调用方仅仅依赖接口调用,且无权选择不同实现. 从使用人员上来说,API 直接被应用开发人员使用.

  • Java SPI简单应用案例详解

    开篇 本文主要谈一下 Java SPI(Service Provider Interface) ,因为最近在看 Dubbo 的相关内容,其中涉及到了 一个概念- Dubbo SPI, 最后又牵扯出来了 JAVA SPI, 所以先从 Java SPI 开整. 正文 平常学习一个知识点,我们的常规做法是: 是什么 有什么用 怎么用 这次我们倒着做,先不谈什么是 SPI 及其作用,来看下如何使用. 使用 1. 创建一个 maven 工程 2. 创建一个接口类以及实现类 // 接口 public int

  • java基础--JDK SPI概述

    目录 JDK SPI是什么 JDK SPI使用说明及示例 SPI在JDBC中的应用 对SPI的理解 JDK SPI是什么 最近工作中听几个同事说了好几次SPI这个名词,虽然和我没关系,但是心里默默想还是学习一下,不然下次和我说到SPI,连是什么都不知道那就尴尬了. 所以SPI是什么呢?SPI全称Service Provider Interface,在Java中还是一个比较重要的概念,是Java提供的一套用来被第三方实现或者扩展的API,或者换句话说,SPI是一种服务发现机制. JDK SPI使用

  • JAVA中的SPI思想介绍

    目录 1. SPI介绍 2. SPI规则 3. SPI案例 3.1 组件的定义 3.2 组件的实现 3.3 组件的选用 4. SPI原理 5. SPI要求 6. SPI应用 总结 1. SPI介绍 SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的接口,其意义在于为某个接口寻找服务的实现,主要应用在框架中用来寻找组件,提高扩展性. 汽车制造是一个比较繁琐的过程,通常的手段是先规定汽车各个零部件的生产规格,各个零部件厂商按照这种规则去生产

  • 在java中使用SPI创建可扩展的应用程序操作

    简介 什么是可扩展的应用程序呢?可扩展的意思是不需要修改原始代码,就可以扩展应用程序的功能.我们将应用程序做成插件或者模块. 这样可以在不修改原应用的基础上,对系统功能进行升级或者定制化. 本文将会向大家介绍如何通过java中的SPI机制实现这种可扩展的应用程序. SPI简介 SPI的全称是Java Service Provider Interface.是java提供的一种服务发现的机制. 通过遵循相应的规则编写应用程序之后,就可以使用ServiceLoader来加载相应的服务了. SPI的实现

  • 深入理解Java中的SPI机制

    本文通过探析JDK提供的,在开源项目中比较常用的Java SPI机制,希望给大家在实际开发实践.学习开源项目提供参考. 1 SPI是什么 SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的API,它可以用来启用框架扩展和替换组件. 整体机制图如下: Java SPI 实际上是"基于接口的编程+策略模式+配置文件"组合实现的动态加载机制. 系统设计的各个抽象,往往有很多不同的实现方案,在面向的对象的设计里,一般推荐模块之间基于接

  • Java中的SPI机制案例分享

    目录 1 简单介绍 2 SPI 案例 3 SPI 的原理剖析 1 简单介绍 当我们封装了一套接口,其它项目想要调用我们的接口只需要引入我们写好的包,但是其它项目如果想要对我们的接口进行扩展,由于接口是被封装在依赖包中的,想要扩展并不容易,这时就需要依赖于Java为我们提供的SPI机制. SPI的全称是Service Provider Interface,服务提供者接口,而与之最接近的概念就是API,全称Application Programming Interface,应用程序编程接口.那么这两

  • 一文带你了解Java中的SPI机制

    目录 1: SPI机制简介 2: SPI原理 3: 使用场景 4: 源码论证 5: 实战 6: 优缺点 6.1 优点 6.2 缺点 1: SPI机制简介 SPI 全称是 Service Provider Interface,是一种 JDK 内置的动态加载实现扩展点的机制,通过 SPI 技术我们可以动态获取接口的实现类,不用自己来创建.这个不是什么特别的技术,只是 一种设计理念. 2: SPI原理 Java SPI 实际上是基于接口的编程+策略模式+配置文件组合实现的动态加载机制. 系统设计的各个

  • Java中的hashcode方法介绍

    哈希表这个数据结构想必大多数人都不陌生,而且在很多地方都会利用到hash表来提高查找效率.在Java的Object类中有一个方法: public native int hashCode(); 根据这个方法的声明可知,该方法返回一个int类型的数值,并且是本地方法,因此在Object类中并没有给出具体的实现. 为何Object类需要这样一个方法?它有什么作用呢?今天我们就来具体探讨一下hashCode方法. 一.hashCode方法的作用 对于包含容器类型的程序设计语言来说,基本上都会涉及到has

  • Java中的递归方法示例介绍

    目录 递归 递归的注意事项: 案例一 递归求阶乘 不使用递归实现阶乘 使用递归实现阶乘 澳大利亚不死神兔(斐波那契数列) 使用数组实现 使用递归实现 总结 递归 方法定义本身调用方法本身的现象叫做递归 在这之前我们学的东西:例如StringBuffer.append().append().append()这个不叫递归.这个叫方法的连续调用Math.max(Math.max(a,b),c)也不是递归,那这些是什么呢?这些是方法的调用. 那什么是递归呢? 举例: 从前有座山,山里有座庙,庙里有个老和

  • 浅析Java中的SPI原理

    在面向对象的程序设计中,模块之间交互采用接口编程,通常情况下调用方不需要知道被调用方的内部实现细节,因为一旦涉及到了具体实现,如果需要换一种实现就需要修改代码,这违反了程序设计的"开闭原则".所以我们一般有两种选择:一种是使用API(Application Programming Interface),另一种是SPI(Service Provider Interface),API通常被应用程序开发人员使用,而SPI通常被框架扩展人员使用. 在进入下面学习之前,我们先来再加深一下API和

  • Java中ArrayList类详细介绍

    Java中ArrayList类详细介绍 ArrayList是一个可变长度数组,它实现了List接口,因此它也可以包含重复元素和Null元素,也可以任意的访问和修改元素,随着向 ArrayList 中不断添加元素,其容量也自动增长.不过ArrayList是非同步(同步的意思是如果多个线程同时访问一个实例,任何一个线程对实例做了修改之后,其他线程所访问到的实例应该是修改过的最新的实例)的, 我们经常使用List list = Collections.synchronizedList(new Arra

  • 重写Java中的equals方法介绍

    Java中,只有8种基本类型不是对象,例如:4种整形类型(byte, short, int,long),2种浮点类型(flout, double),boolean, char不是对象,其他的所有类型,不论是对象数组,列表等都扩展了Object类.了解学习Object中方法的设计原理和实现方式有助于更好的学习理解java语言.下面,我们首先学习一下Object中的equals方法. 判断两个对象相等时,JVM首先查找两个对象的hashCode, 如果两者hashCode不同,则返回false;如果

随机推荐