最近新写了一个中间件「运行时动态日志等级开关」,其中使用Java SPI机制实现了自定义配置中心,保证良好的扩展性。

项目地址,走过路过可以点个star :)

https://github.com/saigu/LogLevelSwitch

在使用过程中,突然发现SPI其实和日常写API接口,然后进行implements实现非常相似,那SPI到底和普通API实现有啥区别呢?

带着这个问题,我们一起来梳理下SPI机制吧。

本文预计阅读时间10分钟,将围绕以下几点展开:

  • 什么是 SPI 机制?
  • SPI 实践案例
  • SPI 和 API 有啥区别?

1、什么是SPI机制?

SPI(Service Provider Interface) 字面意思是服务提供者接口,本质上是一种「服务扩展机制」。

为什么需要这样一种「服务扩展机制」呢?

因为系统里抽象的各个模块,比如日志模块、xml解析模块、jdbc模块等,往往有很多不同的实现方案。

为了满足可拔插的原则,我们一般推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。这就需要一种「服务扩展机制」,然后就有了SPI。

SPI 机制为我们的程序提供拓展功能。而不必将框架的一些实现类写死在代码里面。我们在相应配置文件中定义好某个接口的实现类全限定名,并由服务加载器读取配置文件,加载实现类。这样可以在运行时,动态为接口替换实现类。

最常见的就是Java的SPI机制,另外,还有Dubbo和SpringBoot自定义的SPI机制。

2、SPI实践案例

2.1 业界SPI实践案例

简单了解了SPI的概念,我们看看业界有哪些SPI实践案例,如何利用SPI实现灵活扩展的。

  • JDBC驱动加载

最常见的SPI机制实践案例就是JDBC的驱动加载。利用Java的SPI机制,我们可以根据不同的数据库厂商来引入不同的JDBC驱动包。

  • SpringBoot的SPI机制

用过SpringBoot的同学应该都知道,我们可以在spring.factories中加上我们自定义的自动配置类,这个特性尤其在xxx-starter中应用广泛。

  • Dubbo的SPI机制

Dubbo基本上自身的每个功能点都提供了扩展点,把SPI机制应用的淋漓尽致,比如提供了集群扩展、路由扩展和负载均衡扩展等差不多接近30个扩展点。
如果Dubbo的某个内置实现不符合我们的需求,那么我们只要利用其SPI机制将我们的实现替换掉Dubbo的实现即可。

2.2 在实际项目中如何使用

以上三个例子是业界最常见的SPI机制的实现。下面,来看看我在实际项目中如何利用Java SPI机制实现了自定义配置中心,保证良好的扩展性。

项目地址,走过路过可以点个star :)

https://github.com/saigu/LogLevelSwitch

需求很简单,中间件「运行时动态日志等级开关」需要在应用运行时获取开关状态,然后动态改变应用日志等级。

如何获取开关状态呢?我们一般需要配置中心来进行处理。作为一个开源中间件,使用它的应用可能有自己的不同的配置中心(比如Nacos、Apollo、spring cloud config、自研配置中心等),因此,必须支持自定义配置中心接入。

这时候就需要SPI机制来实现了!

1)定义接口interface

package io.github.saigu.log.level.sw.listener;

public interface ConfigListener<T> {
/**
* 获取初始开关状态
* @return initial context of switch
*/
SwitchContext getInitSwitch(); /**
* 获取变化的配置
* @param changedConfig changed config context
*/
void listenChangedConfig(T changedConfig);
}

2)SPI加载

本项目通过Java SPI实现,不需要依赖额外的组件,通过ServiceLoader来动态加载

public class ChangeListenerFactory {
public static ConfigListener getListener() {
final ServiceLoader<ConfigListener> loader = ServiceLoader
.load(ConfigListener.class);
for (ConfigListener configListener : loader) {
return configListener;
}
throw new IllegalArgumentException("please choose valid listener");
}
}

3)应用自定义配置中心接入

使用这个中间件的应用,只需要三步即可接入自定义配置中心。

  • STEP 1: 应用中pom引入依赖
<dependency>
<groupId>io.github.saigu</groupId>
<artifactId>log-switch-core</artifactId>
<version>1.0.0-beta</version>
</dependency>
  • STEP 2: 构建config Bean
@Configuration
public class LogLevelSwitchConfig {
@Bean
LogLevelSwitch logLevelSwitch() {
return new LogLevelSwitch();
}
}
  • STEP 3: 接入配置中心

声明配置中心的SPI实现。

在resource路径下新建 META-INF/services,创建文件名为
io.github.saigu.log.level.sw.listener.ConfigListener的文件,并写入自定义配置中心的「实现类名」。

3、SPI和API有啥区别?

我们已经介绍了什么是SPI,怎么使用SPI机制,现在,回头来看看一开始提出的问题,SPI和API有啥区别呢?

它们都需要定义接口interface,然后自定义实现类implements,看起来基本一致呀。

区别在哪?各自的使用场景是啥?

别急,我们从头梳理一下。

从「面向接口编程」的思想来看,「调用方」应该通过调用「接口」而不是「具体实现」来处理逻辑。那么,对于「接口」的定义,应该在「调用方」还是「实现方」呢?

理论上来说,会有三种选择:

  • 「接口」定义在「实现方」
  • 「接口」定义在「调用方」
  • 「接口」定义在 独立的包中

1)「接口」定义在「实现方」

先来看看「接口」定义在「实现方」的情况。这个很容易理解,实现方同时提供了「接口」和「实现类」,「调用方」可以引用接口来达到调用某实现类的功能,这就是我们日常使用的API。API的最显著特征就是:

实现和接口在一个包中。自己定义接口,自己实现类。

2)「接口」定义在「调用方」

再来看看「接口」属于「调用方」的情况。这个其实就是SPI机制。以JDBC驱动为例,「调用方」(用户或者说JDK)定义了java.sql.Driver接口,这个接口位于「调用方」JDK的包中,各个数据库厂商实现了这个接口,比如mysql驱动com.mysql.jdbc.Driver。因此,SPI最显著的特征就是:

「接口」在「调用方」的包,「调用方」定义规则,而自定义实现类在「实现方」的包,然后把实现类加载到「调用方」中。

3)「接口」定义在独立的包

最后一种情况,如果一个「接口」在一个上下文是API,在另一个上下文是SPI,那么就可以把「接口」定义在独立的包中。

4、小结

本文介绍了是SPI机制,然后结合业界案例与项目实践来说明SPI的使用场景,最后对Java SPI和API的区别进行了分析。

本文不对SPI原理进行深入解析,下一篇文章会详细分析下Java SPI的实现《大名鼎鼎的Java SPI机制,究竟有没有破坏双亲委派呢?》,应该会挺有意思,欢迎关注。

都看到最后了,原创不易,点个关注,点个赞吧~

文章持续更新,可以微信搜索「阿丸笔记 」第一时间阅读,回复【笔记】获取Canal、MySQL、HBase、JAVA实战笔记,回复【资料】获取一线大厂面试资料。

知识碎片重新梳理,构建Java知识图谱:github.com/saigu/JavaK…(历史文章查阅非常方便)

Java SPI 和 API,傻傻分不清?的更多相关文章

  1. 【jvm】08-垃圾回收器那么多傻傻分不清?

    [jvm]08-垃圾回收器那么多傻傻分不清? 欢迎关注b站账号/公众号[六边形战士夏宁],一个要把各项指标拉满的男人.该文章已在github目录收录. 屏幕前的大帅比和大漂亮如果有帮助到你的话请顺手点 ...

  2. Java:接口和抽象类,傻傻分不清楚?

    01. 来看网络上对接口的一番解释: 接口(英文:Interface),在 Java 编程语言中是一个抽象类型,是抽象方法的集合.一个类通过继承接口的方式,从而来继承接口的抽象方法. 兄弟们,你们怎么 ...

  3. [转帖]十分钟快速理解DPI和PPI,不再傻傻分不清!

    十分钟快速理解DPI和PPI,不再傻傻分不清! https://baijiahao.baidu.com/s?id=1605834796518990333&wfr=spider&for= ...

  4. 学点经济学:M0、M1、M2、M3,傻傻分不清?(转载)

    来源:http://t.10jqka.com.cn/pid_97006727.shtml 学点经济学:M0.M1.M2.M3,傻傻分不清? 25,508人浏览 2018-08-03 11:06 常听人 ...

  5. ASCII、Unicode、UTF-8、UTF-8(without BOM)、UTF-16、UTF-32傻傻分不清

    ASCII.Unicode.UTF-8.UTF-8(without BOM).UTF-16.UTF-32傻傻分不清 目录 ASCII.Unicode.UTF-8.UTF-8(without BOM). ...

  6. MVP MVC MVVM 傻傻分不清

    最近MVC (Model-View-Controller) 和MVVM (Model-View-ViewModel) 在微软圈成为显学,ASP.NET MVC 和WPF 的Prism (MVVM Fr ...

  7. OCA,OCP,OCM傻傻分不清?

    可能大家知道OCA.OCP.OCM的关系是一个比一个难考,一个比一个含金量高,但是你知道具体的考试科目.考试方式.就业形势区别吗?不知道的话这篇通俗易懂的文章会让你一目了然. 区别一:含金量 ■OCA ...

  8. session cookie傻傻分不清

    做了这么多年测试,还是分不清什么是cookie,什么是session?很正常,很多初级开发工程师可能到现在都搞不清什么是session,cookie相对来说会简单很多. 下面这篇文章希望能够帮助大家分 ...

  9. JDBC、ORM、JPA、Spring Data JPA,傻傻分不清楚?一文带你厘清个中曲直,给你个选择SpringDataJPA的理由!

    序言 Spring Data JPA作为Spring Data中对于关系型数据库支持的一种框架技术,属于ORM的一种,通过得当的使用,可以大大简化开发过程中对于数据操作的复杂度. 本文档隶属于< ...

随机推荐

  1. java中的修饰符和基本数据类型

    1.java中的修饰符 java中的修饰符主要是用来对类资源进行一个权限控制,上面表格表现的很清晰,无需多言. 2.java中的基本数据类型 java中的数据类型分为引用类型和基本类型.基本数据类型有 ...

  2. 学习Kvm(一)

     背景介绍 传统数据中心面临的问题: 资源使用率低 资源分配不均 自动化能力差 初始化成本高   云计算: 云计算是一种按使用量付费的模式,这种模式提供可用的.便捷的.按需的网络访问, 进入可配置的计 ...

  3. jsp技术之隐藏域

    隐藏域 hidden:隐藏域属性,不显示到页面上,但是会提交的表单项 注意:表单中增加了一个隐藏域,是用户的id.稍后修改联系人信息,提交表单时需要使用到 <!-- hidden:隐藏域,不显示 ...

  4. 数据结构:DHUOJ 单链表ADT模板应用算法设计:长整数加法运算(使用单链表存储计算结果)

    单链表ADT模板应用算法设计:长整数加法运算(使用单链表存储计算结果) 时间限制: 1S类别: DS:线性表->线性表应用 题目描述: 输入范例: -5345646757684654765867 ...

  5. VueJs单页应用实现微信网页授权及微信分享功能

    在实际开发中,无论是做PC端.WebApp端还是微信公众号等类型的项目的时候,或多或少都会涉及到微信相关的开发,最近公司项目要求实现微信网页授权,并获取微信用户基本信息的功能及微信分享的功能,现在总算 ...

  6. h5 在全屏iphonex中的适配

    iphonex 已经上线有一段时间了,作为业界刘海屏幕第一款机型,导致全屏不能正常的全屏显示了,,所以需要对iphonx 适配,下面就详细说说如何适配 先看一张适配前后的图: iphonex 提供的 ...

  7. pip——重新安装

    原因 没有用管理员的权限安装.导致失败. 并且pip还被卸载了. 使用环境 win10 重新安装pip python -m ensurepip 更新pip 管理员打开cmd python -m pip ...

  8. Django-初见

    目录 安装&启动 HTTP请求URL路由 项目APP 返回 页面内容 给浏览器 路由 路由子表 创建数据库 定义数据库表 创建数据库表 Django Admin 读取数据库数据 过滤条件 对资 ...

  9. Go 1.18泛型的局限性初探

    前言 Go 1.18 版本之后正式引入泛型,它被称作类型参数(type parameters),本文初步介绍 Go 中泛型的使用.长期以来 go 都没有泛型的概念,只有接口 interface 偶尔类 ...

  10. Spring基于注解自动装配

    前面我们介绍Spring IoC装载的时候,使用XML配置这种方法来装配Bean,这种方法可以很直观的看到每个Bean的依赖,但缺点也很明显:写起来非常繁琐,每增加一个组件,就必须把新的Bean配置到 ...