Java加餐篇之Java-SPI机制

最近在准备学习Spring源码,这是一个浩大的工程,所以要考一些知识点,圈起来考的。

SPI是什么

  • SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的API,它可以用来启用框架扩展和替换组件。具体机制如下图所示

1.webp

  • Java SPI 实际上是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制。
  • 系统设计的各个抽象,往往有很多不同的实现方案,在面向的对象的设计里,一般推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,如果需要替换一种实现,就需要修改代码。为了实现在模块装配的时候能不在程序里动态指明,这就需要一种服务发现机制。
    Java SPI就是提供这样的一个机制:为某个接口寻找服务实现的机制。有点类似IOC的思想,就是将装配的控制权移到程序之外,在模块化设计中这个机制尤其重要。所以SPI的核心思想就是解耦

使用场景

  • 调用者根据实际使用需要,启用、扩展、或者替换框架的实现策略
  • 比较常见的例子:
    • 数据库驱动加载接口实现类的加载。JDBC加载不同类型数据库的驱动。
    • 日志门面接口实现类加载。SLF4J加载不同提供商的日志实现类。
    • Spring。Spring中大量使用了SPI,比如:对servlet3.0规范对ServletContainerInitializer的实现、自动类型转换Type Conversion SPI(Converter SPI、Formatter SPI)等
    • Dubbo。Dubbo中也大量使用SPI的方式实现框架的扩展, 不过它对Java提供的原生SPI做了封装,允许用户扩展实现Filter接口。

使用介绍

  • 要使用SPI,必须遵守以下约定
    • 当服务提供者提供了接口的一种具体实现后,在jar包的META-INF/services目录下创建一个以“接口全限定名”为命名的文件,内容为实现类的全限定名
    • 接口实现类所在的jar包放在主程序的classpath中
    • 主程序通过java.util.ServiceLoder动态装载实现模块,它通过扫描META-INF/services目录下的配置文件找到实现类的全限定名,把类加载到JVM
    • SPI的实现类必须携带一个不带参数的构造方法

代码示例

  • 步骤1:定义一个接口、并且编写其实现
  1. public interface IShout {
  2. /**
  3. * say hello
  4. */
  5. void sayHello();
  6. }
  7. public class ShoutOne implements IShout {
  8. /**
  9. * say hello
  10. */
  11. @Override
  12. public void sayHello() {
  13. System.out.println("ShoutOne.sayHello");
  14. }
  15. }
  16. public class ShoutTwo implements IShout {
  17. /**
  18. * say hello
  19. */
  20. @Override
  21. public void sayHello() {
  22. System.out.println("ShoutTwo.sayHello");
  23. }
  24. }
  • 步骤2:在src/main/resources/ 下建立 /META-INF/services 目录, 新增一个以接口全限定类名命名的文件,目录层级如下所示
  1. - src
  2. -main
  3. -resources
  4. - META-INF
  5. - services
  6. - cn.icanci.demo.IShout
  • 文件内容如下
  1. cn.icanci.demo.ShoutOne
  2. cn.icanci.demo.ShoutTwo
  • 步骤3:编写测试main函数进行测试
  1. public class Main {
  2. public static void main(String[] args) {
  3. ServiceLoader<IShout> load = ServiceLoader.load(IShout.class);
  4. Iterator<IShout> iterator = load.iterator();
  5. while (iterator.hasNext()) {
  6. IShout next = iterator.next();
  7. next.sayHello();
  8. }
  9. }
  10. }
  11. // 输出结果如下:
  12. // ShoutOne.sayHello
  13. // ShoutTwo.sayHello

源码解析

  • 其源代码如下所示
  1. package java.util;
  2. import java.io.BufferedReader;
  3. import java.io.IOException;
  4. import java.io.InputStream;
  5. import java.io.InputStreamReader;
  6. import java.net.URL;
  7. import java.security.AccessController;
  8. import java.security.AccessControlContext;
  9. import java.security.PrivilegedAction;
  10. import java.util.ArrayList;
  11. import java.util.Enumeration;
  12. import java.util.Iterator;
  13. import java.util.List;
  14. import java.util.NoSuchElementException;
  15. public final class ServiceLoader<S>
  16. implements Iterable<S>
  17. {
  18. // 需要加载的文件的路径地址
  19. private static final String PREFIX = "META-INF/services/";
  20. // 需要被加载的服务或者类(可以是接口、抽象类、普通类)
  21. private final Class<S> service;
  22. // 类加载器
  23. private final ClassLoader loader;
  24. // 创建ServiceLoader时采用的访问控制上下文
  25. private final AccessControlContext acc;
  26. // 缓存providers,按实例化的顺序排列,也就是被创建的目标实例对象
  27. private LinkedHashMap<String,S> providers = new LinkedHashMap<>();
  28. // 懒查找迭代器
  29. private LazyIterator lookupIterator;
  30. // 重新加载,先清空,再加载
  31. public void reload() {
  32. providers.clear();
  33. lookupIterator = new LazyIterator(service, loader);
  34. }
  35. // 服务加载
  36. private ServiceLoader(Class<S> svc, ClassLoader cl) {
  37. service = Objects.requireNonNull(svc, "Service interface cannot be null");
  38. loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl;
  39. acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null;
  40. reload();
  41. }
  42. // 加载失败
  43. private static void fail(Class<?> service, String msg, Throwable cause)
  44. throws ServiceConfigurationError
  45. {
  46. throw new ServiceConfigurationError(service.getName() + ": " + msg,
  47. cause);
  48. }
  49. // 加载失败
  50. private static void fail(Class<?> service, String msg)
  51. throws ServiceConfigurationError
  52. {
  53. throw new ServiceConfigurationError(service.getName() + ": " + msg);
  54. }
  55. // 加载失败
  56. private static void fail(Class<?> service, URL u, int line, String msg)
  57. throws ServiceConfigurationError
  58. {
  59. fail(service, u + ":" + line + ": " + msg);
  60. }
  61. // 从配置文件中解析一行,并且将改行的名称加载在列表中
  62. private int parseLine(Class<?> service, URL u, BufferedReader r, int lc,
  63. List<String> names)
  64. throws IOException, ServiceConfigurationError
  65. {
  66. String ln = r.readLine();
  67. if (ln == null) {
  68. return -1;
  69. }
  70. int ci = ln.indexOf('#');
  71. if (ci >= 0) ln = ln.substring(0, ci);
  72. ln = ln.trim();
  73. int n = ln.length();
  74. if (n != 0) {
  75. if ((ln.indexOf(' ') >= 0) || (ln.indexOf('\t') >= 0))
  76. fail(service, u, lc, "Illegal configuration-file syntax");
  77. int cp = ln.codePointAt(0);
  78. if (!Character.isJavaIdentifierStart(cp))
  79. fail(service, u, lc, "Illegal provider-class name: " + ln);
  80. for (int i = Character.charCount(cp); i < n; i += Character.charCount(cp)) {
  81. cp = ln.codePointAt(i);
  82. if (!Character.isJavaIdentifierPart(cp) && (cp != '.'))
  83. fail(service, u, lc, "Illegal provider-class name: " + ln);
  84. }
  85. if (!providers.containsKey(ln) && !names.contains(ln))
  86. names.add(ln);
  87. }
  88. return lc + 1;
  89. }
  90. // 通过 URL 加载
  91. private Iterator<String> parse(Class<?> service, URL u)
  92. throws ServiceConfigurationError
  93. {
  94. InputStream in = null;
  95. BufferedReader r = null;
  96. ArrayList<String> names = new ArrayList<>();
  97. try {
  98. in = u.openStream();
  99. r = new BufferedReader(new InputStreamReader(in, "utf-8"));
  100. int lc = 1;
  101. while ((lc = parseLine(service, u, r, lc, names)) >= 0);
  102. } catch (IOException x) {
  103. fail(service, "Error reading configuration file", x);
  104. } finally {
  105. try {
  106. if (r != null) r.close();
  107. if (in != null) in.close();
  108. } catch (IOException y) {
  109. fail(service, "Error closing configuration file", y);
  110. }
  111. }
  112. return names.iterator();
  113. }
  114. // 懒加载迭代器
  115. private class LazyIterator
  116. implements Iterator<S>
  117. {
  118. Class<S> service;
  119. ClassLoader loader;
  120. Enumeration<URL> configs = null;
  121. Iterator<String> pending = null;
  122. String nextName = null;
  123. private LazyIterator(Class<S> service, ClassLoader loader) {
  124. this.service = service;
  125. this.loader = loader;
  126. }
  127. private boolean hasNextService() {
  128. if (nextName != null) {
  129. return true;
  130. }
  131. if (configs == null) {
  132. try {
  133. String fullName = PREFIX + service.getName();
  134. if (loader == null)
  135. configs = ClassLoader.getSystemResources(fullName);
  136. else
  137. configs = loader.getResources(fullName);
  138. } catch (IOException x) {
  139. fail(service, "Error locating configuration files", x);
  140. }
  141. }
  142. while ((pending == null) || !pending.hasNext()) {
  143. if (!configs.hasMoreElements()) {
  144. return false;
  145. }
  146. pending = parse(service, configs.nextElement());
  147. }
  148. nextName = pending.next();
  149. return true;
  150. }
  151. private S nextService() {
  152. if (!hasNextService())
  153. throw new NoSuchElementException();
  154. String cn = nextName;
  155. nextName = null;
  156. Class<?> c = null;
  157. try {
  158. c = Class.forName(cn, false, loader);
  159. } catch (ClassNotFoundException x) {
  160. fail(service,
  161. "Provider " + cn + " not found");
  162. }
  163. if (!service.isAssignableFrom(c)) {
  164. fail(service,
  165. "Provider " + cn + " not a subtype");
  166. }
  167. try {
  168. S p = service.cast(c.newInstance());
  169. providers.put(cn, p);
  170. return p;
  171. } catch (Throwable x) {
  172. fail(service,
  173. "Provider " + cn + " could not be instantiated",
  174. x);
  175. }
  176. throw new Error(); // This cannot happen
  177. }
  178. public boolean hasNext() {
  179. if (acc == null) {
  180. return hasNextService();
  181. } else {
  182. PrivilegedAction<Boolean> action = new PrivilegedAction<Boolean>() {
  183. public Boolean run() { return hasNextService(); }
  184. };
  185. return AccessController.doPrivileged(action, acc);
  186. }
  187. }
  188. public S next() {
  189. if (acc == null) {
  190. return nextService();
  191. } else {
  192. PrivilegedAction<S> action = new PrivilegedAction<S>() {
  193. public S run() { return nextService(); }
  194. };
  195. return AccessController.doPrivileged(action, acc);
  196. }
  197. }
  198. public void remove() {
  199. throw new UnsupportedOperationException();
  200. }
  201. }
  202. // 迭代器
  203. public Iterator<S> iterator() {
  204. return new Iterator<S>() {
  205. Iterator<Map.Entry<String,S>> knownProviders
  206. = providers.entrySet().iterator();
  207. public boolean hasNext() {
  208. if (knownProviders.hasNext())
  209. return true;
  210. return lookupIterator.hasNext();
  211. }
  212. public S next() {
  213. if (knownProviders.hasNext())
  214. return knownProviders.next().getValue();
  215. return lookupIterator.next();
  216. }
  217. public void remove() {
  218. throw new UnsupportedOperationException();
  219. }
  220. };
  221. }
  222. // 加载
  223. public static <S> ServiceLoader<S> load(Class<S> service,
  224. ClassLoader loader)
  225. {
  226. return new ServiceLoader<>(service, loader);
  227. }
  228. // 加载
  229. public static <S> ServiceLoader<S> load(Class<S> service) {
  230. ClassLoader cl = Thread.currentThread().getContextClassLoader();
  231. return ServiceLoader.load(service, cl);
  232. }
  233. // 使用扩展类加载器为给定的服务类型创建一个新的服务加载器
  234. public static <S> ServiceLoader<S> loadInstalled(Class<S> service) {
  235. ClassLoader cl = ClassLoader.getSystemClassLoader();
  236. ClassLoader prev = null;
  237. while (cl != null) {
  238. prev = cl;
  239. cl = cl.getParent();
  240. }
  241. return ServiceLoader.load(service, prev);
  242. }
  243. public String toString() {
  244. return "java.util.ServiceLoader[" + service.getName() + "]";
  245. }
  246. }

实现流程

  • 应用程序调用ServiceLoader.load方法
  • 然后获取类加载器和类,创建ServiceLoader对象
  • 实例化其中的对象,如下
    • loader(ClassLoader类型,类加载器)
    • acc(AccessControlContext类型,访问控制器)
    • providers(LinkedHashMap类型,用于缓存加载成功的类)
    • lookupIterator(实现迭代器功能)
  • 应用程序通过迭代器接口获取对象实例

    • ServiceLoader先判断成员变量providers对象中(LinkedHashMap类型)是否有缓存实例对象,如果有缓存,直接返回。
    • 如果没有缓存,执行类的装载,实现如下
    • 读取META-INF/services/下的配置文件,获得所有能被实例化的类的名称,值得注意的是,ServiceLoader可以跨越jar包获取META-INF下的配置文件,具体加载配置的实现代码如下:

      1. private boolean hasNextService() {
      2. if (nextName != null) {
      3. return true;
      4. }
      5. if (configs == null) {
      6. try {
      7. String fullName = PREFIX + service.getName();
      8. if (loader == null)
      9. configs = ClassLoader.getSystemResources(fullName);
      10. else
      11. configs = loader.getResources(fullName);
      12. } catch (IOException x) {
      13. fail(service, "Error locating configuration files", x);
      14. }
      15. }
      16. while ((pending == null) || !pending.hasNext()) {
      17. if (!configs.hasMoreElements()) {
      18. return false;
      19. }
      20. pending = parse(service, configs.nextElement());
      21. }
      22. nextName = pending.next();
      23. return true;
      24. }
    • 通过反射方法Class.forName()加载类对象,并用instance()方法将类实例化。

    • 把实例化后的类缓存到providers对象中,(LinkedHashMap类型)然后返回实例对象。

总结

优点
  • 使用Java SPI机制的优势是实现解耦,使得第三方服务模块的装配控制的逻辑与调用者的业务代码分离,而不是耦合在一起。应用程序可以根据实际业务情况启用框架扩展或替换框架组件。
  • 相比使用提供接口jar包,供第三方服务模块实现接口的方式,SPI的方式使得源框架,不必关心接口的实现类的路径,可以不用通过下面的方式获取接口实现类:
    • 代码硬编码import 导入实现类
    • 指定类全路径反射获取:例如在JDBC4.0之前,JDBC中获取数据库驱动类需要通过Class.forName(“com.mysql.jdbc.Driver”),类似语句先动态加载数据库相关的驱动,然后再进行获取连接等的操作
    • 第三方服务模块把接口实现类实例注册到指定地方,源框架从该处访问实例
  • 通过SPI的方式,第三方服务模块实现接口后,在第三方的项目代码的META-INF/services目录下的配置文件指定实现类的全路径名,源码框架即可找到实现类

缺点
  • 虽然ServiceLoader也算是使用的延迟加载,但是基本只能通过遍历全部获取,也就是接口的实现类全部加载并实例化一遍。如果你并不想用某些实现类,它也被加载并实例化了,这就造成了浪费。获取某个实现类的方式不够灵活,只能通过Iterator形式获取,不能根据某个参数来获取对应的实现类。
  • 多个并发多线程使用ServiceLoader类的实例是不安全的。