一、java spi
对于java spi大部分开发者都不会陌生,我们使用的第三方框架的配置如:jdbc、日志、spring、微服务框架等。spi全称:Service Provider Interface,实现了模块间的解耦,实现可拔插。
java spi的约束如下:
- 在META-INF/services/目录中创建以接口全限定名命名的文件该文件内容为Api具体实现类的全限定名
- 使用ServiceLoader类动态加载META-INF中的实现类
- 如SPI的实现类为Jar则需要放在主程序classPath中
- Api具体实现类必须有一个不带参数的构造方法
我们用jdbc driver 来看下具体的实现,对于不同的数据库驱动是不一样的,如mysql、oracle等。那么每个对应的驱动实现就是从在META-INF/services/下面去寻找对应的驱动。
我们看下DriverManager这个类,
DriverManager#loadInitialDrivers
private static void loadInitialDrivers() {
String drivers;
//代码省略。。。
AccessController.doPrivileged(new PrivilegedAction<Void>() {
public Void run() {
//通过ServiceLoader加载 不同厂商的驱动(myaql、oracle等)
ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
Iterator<Driver> driversIterator = loadedDrivers.iterator();
//代码省略。。。
我们在看mysql-connector-java-.jar下面META-INF/services/是否有对应的实现。
我们再理一下,首先提供接口,不同的服务厂商或者第三方做接口实现。这样不用在JDK里实现Driver实现类的硬编码,然后每次使用JDK里的DriverManager 类时,都会自动去发现Driver类的实现类,并根据这个实现类来做数据库连接。又可以满足不同的产商实现各不相同,但对外暴露一样的接口。使用方只要按照JDK的标准方法来调用即可,即实现了接口的可拔插。
二、其中spring的SPI模式,
通过SpringFactoriesLoader代替JDK中ServiceLoader,通过META-INF/spring.factories文件代替META-INF/service目录下的描述文件来实现SPI。一般我们较多是spring 中的数据源、第三方框架等。
这里就不做叙述了。有兴趣的自己可以参照数据源加载(DataSourceAutoConfiguration)。