![ca9f288d7e2ed94758c3efdb8ecdb099.png](https://img-blog.csdnimg.cn/img_convert/ca9f288d7e2ed94758c3efdb8ecdb099.png)
SPI简介
SPI
全称是Service Provider Interface
,可称之为“服务供给接口”,是JDK
内置的一种服务提供发现机制,其实这个确切地说是一种思想,并不是Java
独有的专利。谈到SPI
不得不提的一个概念就是“面向接口编程”,SPI
通过事先定义的接口来找到其所有的接口实现类(服务),而后通过反射逐个实例化对象,这便是SPI
的本质。
为什么需要SPI
SPI
也被叫做服务发现机制,之所以需要服务发现,是因为接口实现的不确定性。就拿java.sql.Driver
来说,如果你没有引入类似于MySQL
这样的实现,你不会知道它的实现类:
![1db5b4e1e8ef435438c01a00ff7d8797.png](https://img-blog.csdnimg.cn/img_convert/1db5b4e1e8ef435438c01a00ff7d8797.png)
JDK
就只是提供了这样一个接口,却并不打算给它实现,为什么?因为现实中的数据库厂商实在是太多了:MySQL
、SQL Server
、Oracle
等等,如果都要由Java
官方实现和维护,那工程量实在太大!除此之外还有众多因素制约。
我们知道Java
中要通过反射实例化一个类是需要拿到具体类的全限定类名称的,像这样:
class Sample {
public static void main(String... args) {
Class<?> serviceImpl = Class.forName("com.tinysand.example.ServiceImpl");
serviceImpl.getDeclaredConstructor().newInstance();
}
}
然而对于类似java.sql.Driver
接口这样的情况你完全不可能知道它实现类的全限定类名。
延迟加载与可扩展性
SPI
也是一种将实现类的加载延迟的一种机制,我们可以面向接口编程,但免不了要提供一种“在运行时也能正确找到并实例化实现类的途径”,上面的Class.forName("com.tinysand.example.ServiceImpl")
不失为一种途径&#x