本文经过探析JDK提供的,在开源项目中比较经常使用的Java SPI机制,但愿给你们在实际开发实践、学习开源项目提供参考。java
SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的API,它能够用来启用框架扩展和替换组件。数据库
总体机制图以下:编程
Java SPI 其实是“ 基于接口的编程+策略模式+配置文件 ”组合实现的动态加载机制。缓存
系统设计的各个抽象,每每有不少不一样的实现方案,在面向的对象的设计里,通常推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,若是须要替换一种实现,就须要修改代码。为了实如今模块装配的时候能不在程序里动态指明,这就须要一种服务发现机制。 Java SPI就是提供这样的一个机制:为某个接口寻找服务实现的机制。有点相似IOC的思想,就是将装配的控制权移到程序以外,在模块化设计中这个机制尤为重要。因此SPI的核心思想就是 解耦 。安全
归纳地说,适用于: 调用者根据实际使用须要,启用、扩展、或者替换框架的实现策略多线程
比较常见的例子:并发
要使用Java SPI,须要遵循以下约定:框架
步骤一、定义一组接口 (假设是org.foo.demo.IShout),并写出接口的一个或多个实现,(假设是org.foo.demo.animal.Dog、org.foo.demo.animal.Cat)。ide
public interface IShout { void shout(); } public class Cat implements IShout { @Override public void shout() { System.out.println("miao miao"); } } public class Dog implements IShout { @Override public void shout() { System.out.println("wang wang"); } } 复制代码
步骤二、在 src/main/resources/ 下创建 /META-INF/services 目录, 新增一个以接口命名的文件 (org.foo.demo.IShout文件),内容是要应用的实现类(这里是org.foo.demo.animal.Dog和org.foo.demo.animal.Cat,每行一个类)。模块化
文件位置
- src -main -resources - META-INF - services - org.foo.demo.IShout 复制代码
文件内容
org.foo.demo.animal.Dog org.foo.demo.animal.Cat 复制代码
步骤三、使用 ServiceLoader 来加载配置文件中指定的实现。
public class SPIMain { public static void main(String[] args) { ServiceLoader<IShout> shouts = ServiceLoader.load(IShout.class); for (IShout s : shouts) { s.shout(); } } } 复制代码
代码输出:
wang wang miao miao 复制代码
首先看ServiceLoader类的签名类的成员变量:
public final class ServiceLoader<S> implements Iterable<S>{ private static final String PREFIX = "META-INF/services/"; // 表明被加载的类或者接口 private final Class<S> service; // 用于定位,加载和实例化providers的类加载器 private final ClassLoader loader; // 建立ServiceLoader时采用的访问控制上下文 private final AccessControlContext acc; // 缓存providers,按实例化的顺序排列 private LinkedHashMap<String,S> providers = new LinkedHashMap<>(); // 懒查找迭代器 private LazyIterator lookupIterator; ...... } 复制代码
参考具体ServiceLoader具体源码,代码量很少,加上注释一共587行,梳理了一下,实现的流程以下:
1 应用程序调用ServiceLoader.load方法 ServiceLoader.load方法内先建立一个新的ServiceLoader,并实例化该类中的成员变量,包括:
2 应用程序经过迭代器接口获取对象实例 ServiceLoader先判断成员变量providers对象中(LinkedHashMap<String,S>类型)是否有缓存实例对象,若是有缓存,直接返回。 若是没有缓存,执行类的装载,实现以下:
(1) 读取META-INF/services/下的配置文件,得到全部能被实例化的类的名称,值得注意的是,ServiceLoader 能够跨越jar包获取META-INF下的配置文件 ,具体加载配置的实现代码以下:
try { String fullName = PREFIX + service.getName(); if (loader == null) configs = ClassLoader.getSystemResources(fullName); else configs = loader.getResources(fullName); } catch (IOException x) { fail(service, "Error locating configuration files", x); } 复制代码
优势: 使用Java SPI机制的优点是实现解耦,使得第三方服务模块的装配控制的逻辑与调用者的业务代码分离,而不是耦合在一块儿。应用程序能够根据实际业务状况启用框架扩展或替换框架组件。