一.ServiceLoader原理
上一节我们介绍了JDBC是如何通过暴露接口给客户方去扩展的。服务方有几个比较麻烦的地方。
- 需要提供一个驱动管理程序去管理所有驱动
- 在用户使用时需要自行使用Class.forName去加载驱动,用户需要去查询具体驱动的全限定名
什么是ServiceLoader
ServiceLoader是在JDK1.6时引入的一个新特性,用于实现SPI。客户方只需要在扩展程序的资源目录META-INF/services中放置配置文件,然后在服务方调用ServiceLoader.load(XXXInterface.class)即可加载扩展接口。
public final class ServiceLoader<S>
implements Iterable<S>{
private static final String PREFIX = “META-INF/services/“;
// Cached providers, in instantiation order
private LinkedHashMap<String,S> providers = new LinkedHashMap<>();
}
过程如下:
1.调用load方法传入class对象
2.获取class对象的全限定名name
3.在当前线程类加载器的类路径下的META-INF/services目录中读取名称为name的文件
4.逐行读取文件中的类全限定名,通过反射创建该类对象并放入providers中
实际用法
public class Main {
public static void main(String[] args) {
ServiceLoader<MyDriver> myDrivers = ServiceLoader.load(MyDriver.class);
for (MyDriver myDriver : myDrivers) {
myDriver.getConnection("mysql://127.0.0.1");
}
}
}
二.ServiceLoader的应用
SpringBoot Starters启动器
Starters可以理解为启动器,它包含了一系列可以集成到应用里面的依赖包,无论你想使用哪种框架技术,引一个starter就能很好地让Spring和其他框架进行融合,不需要额外的整合工作。如你想使用Spring JPA访问数据库,只要加入spring-boot-starter-data-jpa启动器依赖就能使用了。
Spring是怎么设计starter的
周边博客有很多介绍如何自定义starter,在最后一步都需要在资源目录的META-INF下增加一个配置文件spring.factories
SpringBoot在启动时会扫描引入的jar包中所有spring.factories资源文件,读取其中的AutoConfiguration自动配置类将启动器的类注册到spring容器中进行管理。
该机制就是基于SPI的思想,其核心类SpringFactoriesLoader与ServiceLoader相似