java实践SPI机制及浅析源码

小灰灰 2023-02-28 15:28 92阅读 0赞

1.概念

正式步入今天的核心内容之前,溪源先给大家介绍一下关于SPI机制的相关概念,最后会提供实践源代码。

SPI即Service Provider Interface,属于JDK内置的一种动态的服务提供发现机制,可以理解为运行时动态加载接口的实现类。更甚至,大家可以将SPI机制与设计模式中的策略模式建立联系。

  • SPI机制:
    在这里插入图片描述

从上图中理解SPI机制:标准化接口+策略模式+配置文件

SPI机制核心思想:系统设计的各个抽象,往往有很多不同的实现方案,在面向的对象的设计里,一般推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,如果需要替换一种实现,就需要修改代码。为了实现在模块装配的时候能不在程序里动态指明,这就需要一种服务发现机制

  • 使用场景:

    1. 数据库驱动加载:面对不同厂商的数据库,JDBC需要加载不同类型的数据库驱动;
    2. 日志接口实现:SLF4J加载不同日志实现类;
    3. 溪源在实际开发中也使用了SPI机制:面对不同仪器平台的结果文件上传需要解析具体的结果,文件不同,解析逻辑不同,因此采用SPI机制能够解耦和降低维护成本;
  • SPI机制使用约定:

    从上面的图中,我们可以清晰的知道SPI的三部分:接口+实现类+配置文件;因此,项目中若要利用SPI机制,则需要遵循以下约定:

    1. 当服务提供者提供了接口的一种具体实现后,在jar包的META-INF/services目录下创建一个以“接口全限定名”为命名的文件,内容为实现类的全限定名。
    2. 主程序通过java.util.ServiceLoder动态装载实现模块,它通过扫描META-INF/services目录下的配置文件找到实现类的全限定名,把类加载到JVM;

2.实践

整体包结构如图:
在这里插入图片描述

  • 新建标准化接口:

    public interface SayService {

    1. void say(String word);

    }

  • 建立两个实现类

    @Service
    public class ASayServiceImpl implements SayService {

    1. @Override
    2. public void say(String word) {
    3. System.out.println(word + " A say: I am a boy");
    4. }

    }

  1. @Service
  2. public class BSayServiceImpl implements SayService {
  3. @Override
  4. public void say(String word) {
  5. System.out.println(word + " B say: I am a girl");
  6. }
  7. }
  • 新建META-INF/services目录和配置文件(以接口全限定名)

配置文件内容为实现类全限定名

  1. com.qxy.spi.impl.ASayServiceImpl
  2. com.qxy.spi.impl.BSayServiceImpl
  • 单测

    @SpringBootTest
    @RunWith(SpringRunner.class)
    public class SpiTest {

    1. static ServiceLoader<SayService> services = ServiceLoader.load(SayService.class);
    2. @Test
    3. public void test1() {
    4. for (SayService sayService : services) {
    5. sayService.say("Hello");
    6. }
    7. }

    }

  • 结果

    Hello A say: I am a boy
    Hello B say: I am a girl

3.源码

源码主要加载流程如下:

  • 应用程序调用ServiceLoader.load方法 ServiceLoader.load方法内先创建一个新的ServiceLoader,并实例化该类中的成员变量;
  1. loader(ClassLoader类型,类加载器)
  2. acc(AccessControlContext类型,访问控制器)
  3. providers(LinkedHashMap类型,用于缓存加载成功的类)
  4. lookupIterator(实现迭代器功能)

    • 应用程序通过迭代器接口获取对象实例 ServiceLoader先判断成员变量providers对象中(LinkedHashMap类型)是否有缓存实例对象,如果有缓存,直接返回。如果没有缓存,执行类的装载。
  5. 读取META-INF/services/下的配置文件,获得所有能被实例化的类的名称,值得注意的是,ServiceLoader可以跨越jar包获取META-INF下的配置文件;

  6. 通过反射方法Class.forName()加载类对象,并用instance()方法将类实例化。
  7. 把实例化后的类缓存到providers对象中,(LinkedHashMap类型) 然后返回实例对象。

    public final class ServiceLoader

    1. implements Iterable<S>

    {

    1. // 加载具体实现类信息的前缀
    2. private static final String PREFIX = "META-INF/services/";
    3. // 需要加载的接口
    4. // The class or interface representing the service being loaded
    5. private final Class<S> service;
    6. // 用于加载的类加载器
    7. // The class loader used to locate, load, and instantiate providers
    8. private final ClassLoader loader;
    9. // 创建ServiceLoader时采用的访问控制上下文
    10. // The access control context taken when the ServiceLoader is created
    11. private final AccessControlContext acc;
    12. // 用于缓存已经加载的接口实现类,其中key为实现类的完整类名
    13. // Cached providers, in instantiation order
    14. private LinkedHashMap<String,S> providers = new LinkedHashMap<>();
    15. // 用于延迟加载接口的实现类
    16. // The current lazy-lookup iterator
    17. private LazyIterator lookupIterator;
  1. public void reload() {
  2. providers.clear();
  3. lookupIterator = new LazyIterator(service, loader);
  4. }
  5. private ServiceLoader(Class<S> svc, ClassLoader cl) {
  6. service = Objects.requireNonNull(svc, "Service interface cannot be null");
  7. loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl;
  8. acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null;
  9. reload();
  10. }
  11. private static void fail(Class<?> service, String msg, Throwable cause)
  12. throws ServiceConfigurationError
  13. {
  14. throw new ServiceConfigurationError(service.getName() + ": " + msg,
  15. cause);
  16. }
  17. private static void fail(Class<?> service, String msg)
  18. throws ServiceConfigurationError
  19. {
  20. throw new ServiceConfigurationError(service.getName() + ": " + msg);
  21. }
  22. private static void fail(Class<?> service, URL u, int line, String msg)
  23. throws ServiceConfigurationError
  24. {
  25. fail(service, u + ":" + line + ": " + msg);
  26. }
  27. // Parse a single line from the given configuration file, adding the name
  28. // on the line to the names list.
  29. //具体解析资源文件中的每一行内容
  30. private int parseLine(Class<?> service, URL u, BufferedReader r, int lc,
  31. List<String> names)
  32. throws IOException, ServiceConfigurationError
  33. {
  34. String ln = r.readLine();
  35. if (ln == null) {
  36. //-1表示解析完成
  37. return -1;
  38. }
  39. // 如果存在'#'字符,截取第一个'#'字符串之前的内容,'#'字符之后的属于注释内容
  40. int ci = ln.indexOf('#');
  41. if (ci >= 0) ln = ln.substring(0, ci);
  42. ln = ln.trim();
  43. int n = ln.length();
  44. if (n != 0) {
  45. //不合法的标识:' '、'\t'
  46. if ((ln.indexOf(' ') >= 0) || (ln.indexOf('\t') >= 0))
  47. fail(service, u, lc, "Illegal configuration-file syntax");
  48. int cp = ln.codePointAt(0);
  49. //判断第一个 char 是否一个合法的 Java 起始标识符
  50. if (!Character.isJavaIdentifierStart(cp))
  51. fail(service, u, lc, "Illegal provider-class name: " + ln);
  52. //判断所有其他字符串是否属于合法的Java标识符
  53. for (int i = Character.charCount(cp); i < n; i += Character.charCount(cp)) {
  54. cp = ln.codePointAt(i);
  55. if (!Character.isJavaIdentifierPart(cp) && (cp != '.'))
  56. fail(service, u, lc, "Illegal provider-class name: " + ln);
  57. }
  58. //不存在则缓存
  59. if (!providers.containsKey(ln) && !names.contains(ln))
  60. names.add(ln);
  61. }
  62. return lc + 1;
  63. }
  64. private Iterator<String> parse(Class<?> service, URL u)
  65. throws ServiceConfigurationError
  66. {
  67. InputStream in = null;
  68. BufferedReader r = null;
  69. ArrayList<String> names = new ArrayList<>();
  70. try {
  71. in = u.openStream();
  72. r = new BufferedReader(new InputStreamReader(in, "utf-8"));
  73. int lc = 1;
  74. while ((lc = parseLine(service, u, r, lc, names)) >= 0);
  75. } catch (IOException x) {
  76. fail(service, "Error reading configuration file", x);
  77. } finally {
  78. try {
  79. if (r != null) r.close();
  80. if (in != null) in.close();
  81. } catch (IOException y) {
  82. fail(service, "Error closing configuration file", y);
  83. }
  84. }
  85. return names.iterator();
  86. }
  87. // Private inner class implementing fully-lazy provider lookup
  88. //
  89. private class LazyIterator
  90. implements Iterator<S>
  91. {
  92. Class<S> service;
  93. ClassLoader loader;
  94. // 加载资源的URL集合
  95. Enumeration<URL> configs = null;
  96. // 需加载的实现类的全限定类名的集合
  97. Iterator<String> pending = null;
  98. // 下一个需要加载的实现类的全限定类名
  99. String nextName = null;
  100. private LazyIterator(Class<S> service, ClassLoader loader) {
  101. this.service = service;
  102. this.loader = loader;
  103. }
  104. private boolean hasNextService() {
  105. if (nextName != null) {
  106. return true;
  107. }
  108. if (configs == null) {
  109. try {
  110. // 资源名称,META-INF/services + 全限定名
  111. String fullName = PREFIX + service.getName();
  112. if (loader == null)
  113. configs = ClassLoader.getSystemResources(fullName);
  114. else
  115. configs = loader.getResources(fullName);
  116. } catch (IOException x) {
  117. fail(service, "Error locating configuration files", x);
  118. }
  119. }
  120. // 从资源中解析出需要加载的所有实现类的全限定名
  121. while ((pending == null) || !pending.hasNext()) {
  122. if (!configs.hasMoreElements()) {
  123. return false;
  124. }
  125. pending = parse(service, configs.nextElement());
  126. }
  127. //下一个需要加载的实现类全限定名
  128. nextName = pending.next();
  129. return true;
  130. }
  131. private S nextService() {
  132. if (!hasNextService())
  133. throw new NoSuchElementException();
  134. String cn = nextName;
  135. nextName = null;
  136. Class<?> c = null;
  137. try {
  138. //反射构造Class实例
  139. c = Class.forName(cn, false, loader);
  140. } catch (ClassNotFoundException x) {
  141. fail(service,
  142. "Provider " + cn + " not found");
  143. }
  144. // 类型判断,校验实现类必须与当前加载的类/接口的关系是派生或相同,否则抛出异常终止
  145. if (!service.isAssignableFrom(c)) {
  146. fail(service,
  147. "Provider " + cn + " not a subtype");
  148. }
  149. try {
  150. //强转
  151. S p = service.cast(c.newInstance());
  152. // 实例完成,添加缓存,Key:实现类全限定类名,Value:实现类实例
  153. providers.put(cn, p);
  154. return p;
  155. } catch (Throwable x) {
  156. fail(service,
  157. "Provider " + cn + " could not be instantiated",
  158. x);
  159. }
  160. throw new Error(); // This cannot happen
  161. }
  162. public boolean hasNext() {
  163. if (acc == null) {
  164. return hasNextService();
  165. } else {
  166. PrivilegedAction<Boolean> action = new PrivilegedAction<Boolean>() {
  167. public Boolean run() { return hasNextService(); }
  168. };
  169. return AccessController.doPrivileged(action, acc);
  170. }
  171. }
  172. public S next() {
  173. if (acc == null) {
  174. return nextService();
  175. } else {
  176. PrivilegedAction<S> action = new PrivilegedAction<S>() {
  177. public S run() { return nextService(); }
  178. };
  179. return AccessController.doPrivileged(action, acc);
  180. }
  181. }
  182. public void remove() {
  183. throw new UnsupportedOperationException();
  184. }
  185. }
  186. public Iterator<S> iterator() {
  187. return new Iterator<S>() {
  188. Iterator<Map.Entry<String,S>> knownProviders
  189. = providers.entrySet().iterator();
  190. public boolean hasNext() {
  191. if (knownProviders.hasNext())
  192. return true;
  193. return lookupIterator.hasNext();
  194. }
  195. public S next() {
  196. if (knownProviders.hasNext())
  197. return knownProviders.next().getValue();
  198. return lookupIterator.next();
  199. }
  200. public void remove() {
  201. throw new UnsupportedOperationException();
  202. }
  203. };
  204. }
  205. public static <S> ServiceLoader<S> load(Class<S> service,
  206. ClassLoader loader)
  207. {
  208. // 返回ServiceLoader的实例
  209. return new ServiceLoader<>(service, loader);
  210. }
  211. public static <S> ServiceLoader<S> loadInstalled(Class<S> service) {
  212. ClassLoader cl = ClassLoader.getSystemClassLoader();
  213. ClassLoader prev = null;
  214. while (cl != null) {
  215. prev = cl;
  216. cl = cl.getParent();
  217. }
  218. return ServiceLoader.load(service, prev);
  219. }
  220. public String toString() {
  221. return "java.util.ServiceLoader[" + service.getName() + "]";
  222. }
  223. }

4.总结

SPI机制在实际开发中使用得场景也有很多。特别是统一标准的不同厂商实现,溪源也正是利用SPI机制(但略做改进,避免过多加载资源浪费)实现不同技术平台的结果文件解析需求。

优点

使用Java SPI机制的优势是实现解耦,使得第三方服务模块的装配控制的逻辑与调用者的业务代码分离,而不是耦合在一起。应用程序可以根据实际业务情况启用框架扩展或替换框架组件。

缺点

虽然ServiceLoader也算是使用的延迟加载,但是基本只能通过遍历全部获取,也就是接口的实现类全部加载并实例化一遍。如果你并不想用某些实现类,它也被加载并实例化了,这就造成了浪费。

源码传送门:SPI Service

发表评论

表情:
评论列表 (有 0 条评论,92人围观)

还没有评论,来说两句吧...

相关阅读

    相关 dubbo解析-SPI机制

     架构体系   框架介绍   概述   Dubbo是阿里巴巴公司开源的一个高性能优秀的服务框架,使得应用可通过高性能的 RPC 实现服务的输出和输入功能,可以和 Spri