文档章节

浅谈 Java SPI 机制

nimo10050
 nimo10050
发布于 03/22 14:10
字数 1013
阅读 85
收藏 0

开篇

本文主要谈一下 Java SPI(Service Provider Interface) ,因为最近在看 Dubbo 的相关内容,其中涉及到了 一个概念- Dubbo SPI, 最后又牵扯出来了 JAVA SPI, 所以先从 Java SPI 开整。

正文

平常学习一个知识点,我们的常规做法是:

  • 是什么
  • 有什么用
  • 怎么用

这次我们倒着做,先不谈什么是 SPI 及其作用,来看下如何使用。

使用

1. 创建一个 maven 工程

在这里插入图片描述

2. 创建一个接口类以及实现类


// 接口
public interface HelloService {

    void sayHello();

}

// 实现类 1
public class HelloServiceImpl1 implements HelloService {
    @Override
    public void sayHello() {
        System.out.println("hello impl1");
    }
}

// 实现类 2
public class HelloServiceImpl2 implements HelloService {
    @Override
    public void sayHello() {
        System.out.println("hello impl2");
    }
}

3. 创建一个 META-INF/services 文件夹,并添加一个文件

在 classpath 下面创建一个META-INF/services目录,再在下面创建一个 以接口类全路径名 命名的文件 即:com.nimo.spidemo.service.HelloService

4. 在第三步创建的文件中写入如下内容

写入两个实现类的全路径名

com.nimo.spidemo.service.impl.HelloServiceImpl1
com.nimo.spidemo.service.impl.HelloServiceImpl2

5. 启动函数

public class App {

    public static void main(String[] args) throws ClassNotFoundException, SQLException {
        // 方式 一
        Iterator<HelloService> providers = Service.providers(HelloService.class);
        while(providers.hasNext()) {
            HelloService ser = providers.next();
            ser.sayHello();
        }

        System.out.println("-----------------分割线---------------");
		// 方式 二
        ServiceLoader<HelloService> load = ServiceLoader.load(HelloService.class);
        Iterator<HelloService> iterator = load.iterator();
        while(iterator.hasNext()) {
            HelloService ser = iterator.next();
            ser.sayHello();
        }
    }

}

运行结果如下:

hello impl1
hello impl2
-----------------分割线---------------
hello impl1
hello impl2

使用要素

针对上面的 demo,可以看出使用 Java SPI 的几个关键要素:

  1. 接口类,比如 HelloService
  2. 对应接口的实现类 实现类必须携带一个不带参数的构造方法
  3. 文件夹 META-INF/services 放置 classpath 目录下
  4. 以“接口全限定名”命名的文件
  5. 文件内容为接口实现类的全路径

主程序通过java.util.ServiceLoder 扫描META-INF/services下的配置文件,然后会找到实现类的全限定名,最后把类加载到JVM;

SPI 的作用

从上面的 demo 中可以看到,Java SPI 就是把某个接口的 指定实现类 (通过在指定文件配置的方式)给实例化出来了。

准确+官方的说:

SPI 是一种将服务接口与服务实现分离以达到解耦、大大提升了程序可扩展性的机制。引入服务提供者就是引入了 spi 接口的实现者,通过本地的注册发现获取到具体的实现类,轻松可插拔。

~~~ 如果还是不懂就接着往下看⬇️

SPI 的应用场景

一个典型的案例就是 jdbc 。

数据库各大厂商(如Mysql、Oracle)会根据一个统一的接口规范( java.sql.Driver )开发各自的驱动实现逻辑。 客户端使用 jdbc 时不需要去改变代码,直接引入不同的 spi 接口服务即可。 例如 : Mysql 是 com.mysql.jdbc.Drive Oracle 是 oracle.jdbc.driver.OracleDriver

一段熟悉的代码:

使用操作 mysql 数据库的前置工作:

 //1.加载驱动程序
Class.forName("com.mysql.jdbc.Driver");
//2. 获得数据库连接
Connection conn = DriverManager.getConnection(URL, USER, PASSWORD);

当我们需要切到 oracle 数据库时,更改驱动为 oracle.jdbc.driver.OracleDriver, 最后修改连接信息【用户名,密码等】即可。

总结

Java SPI 是一种服务发现机制。它通过在 classPath 路径下的 META-INF/services 文件夹查找文件,自动加载文件里所定义的类。

它的核心关键作用就是 扩展

其他应用场景:

  • 日志门面接口实现类加载

    SLF4J加载不同提供商的日志实现类

  • Spring

    Spring中大量使用了 SPI,比如:对 servlet3.0 规范 ServletContainerInitializer 的实现、自动类型转换Type Conversion SPI(Converter SPI、Formatter SPI)等

  • Dubbo

    Dubbo中也大量使用SPI的方式实现框架的扩展, 不过它对Java提供的原生SPI做了封装,允许用户扩展实现Filter接口

其中 Dubbo 中的 SPI 是接下来研究的重点。

© 著作权归作者所有

nimo10050
粉丝 5
博文 61
码字总数 64507
作品 0
杭州
程序员
私信 提问
加载中

评论(0)

深入理解Java类加载器(ClassLoader)

深入理解Java类加载器(ClassLoader) Java学习记录--委派模型与类加载器 关于Java类加载双亲委派机制的思考(附一道面试题) 真正理解线程上下文类加载器(多案例分析) [jvm解析系列][十]...

dai.sp
2018/09/12
0
0
Java类加载器ClassLoader

JAVA类装载方式,有两种: 1.隐式装载, 程序在运行过程中当碰到通过new 等方式生成对象时,隐式调用类装载器加载对应的类到jvm中。 2.显式装载, 通过class.forname()等方法,显式加载需要的...

兴国First
2018/10/14
0
0
JDK源码之ClassLoader

Java类加载器ClassLoader总结 JAVA类装载方式,有两种: 1.隐式装载, 程序在运行过程中当碰到通过new 等方式生成对象时,隐式调用类装载器加载对应的类到jvm中。 2.显式装载, 通过class.for...

村长大神
2014/03/27
28
0
Java类加载器ClassLoader总结

JAVA类装载方式,有两种: 1.隐式装载, 程序在运行过程中当碰到通过new 等方式生成对象时,隐式调用类装载器加载对应的类到jvm中。 2.显式装载, 通过class.forname()等方法,显式加载需要的...

亭子happy
2019/02/28
30
0
聊聊Dubbo - Dubbo可扩展机制实战

1. Dubbo的扩展机制 在Dubbo的官网上,Dubbo描述自己是一个高性能的RPC框架。今天我想聊聊Dubbo的另一个很棒的特性, 就是它的可扩展性。 如同罗马不是一天建成的,任何系统都一定是从小系统不...

中间件小哥
2018/05/29
0
0

没有更多内容

加载失败,请刷新页面

加载更多

java面向对象编程3(8) 之 BeanUtils工具类

BeanUtils 一. 引言&概述&入门使用 BeanUtils可以让我们更方便快捷地 封装JavaBean数据。 例如可以用在带参构造, Set方法 , 反射。 入门使用: 导入jar包: public static void main(String[]...

煌sir
23分钟前
9
0
python 文件读写with open模式

模式 可做操作 若文件不存在 是否覆盖 r 只能读 报错 - r+ 可读可写 报错 是 w 只能写 创建 是 w+  可读可写 创建 是 a   只能写 创建 否,追加写 a+ 可读可写 创建 否,追加写 例子: wi...

好大一条鱼
23分钟前
9
0
动画:一篇文章快速学会桶排序

内容介绍 桶排序简介 前面学过计数排序,计数排序是一个非基于比较的排序算法,它的优势在于在对一定范围内的整数排序时,它的复杂度为Ο(n+k)。 所谓桶就是存放多个数据的容器,桶排序也是非...

表哥动画学编程
25分钟前
17
0
五行代码实现千万类别分类网络,飞桨大规模分类库揭秘

“桃花一簇无开主,可爱深红爱浅红。 黄四娘家花满蹊,千朵万朵压枝低。 留连戏蝶时时舞,自在娇莺恰恰啼。” 春天来了,经过一个冬天的“窖藏”,按耐不住的小伙伴纷纷行动了起来,踏一踏满...

飞桨PaddlePaddle
36分钟前
7
0
hadoop8天课程——第三天,MapReduce详解

MR擅长处理离线的文本文件。MR+yarn将简单的运算逻辑很方便的拓展到海量数据背景下运行分布式程序逻辑。 如果自己来写分布式程序的话,那么首先会面临一个问题,如何将我的计算逻辑分发到其他...

jingmamour
43分钟前
13
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部