本文经过探析JDK提供的,在开源项目中比较经常使用的Java SPI机制,但愿给你们在实际开发实践、学习开源项目提供参考。java
SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的API,它能够用来启用框架扩展和替换组件。面试
总体机制图以下:数据库
Java SPI 其实是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制。编程
系统设计的各个抽象,每每有不少不一样的实现方案,在面向的对象的设计里,通常推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,若是须要替换一种实现,就须要修改代码。为了实如今模块装配的时候能不在程序里动态指明,这就须要一种服务发现机制。
Java SPI就是提供这样的一个机制:为某个接口寻找服务实现的机制。有点相似IOC的思想,就是将装配的控制权移到程序以外,在模块化设计中这个机制尤为重要。因此SPI的核心思想就是解耦。设计模式
归纳地说,适用于:调用者根据实际使用须要,启用、扩展、或者替换框架的实现策略缓存
比较常见的例子:安全
要使用Java SPI,须要遵循以下约定:多线程
步骤1、定义一组接口 (假设是org.foo.demo.IShout),并写出接口的一个或多个实现,(假设是org.foo.demo.animal.Dog、org.foo.demo.animal.Cat)。并发
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"); } }
步骤2、在 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
步骤3、使用 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行,梳理了一下,实现的流程以下:
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机制的优点是实现解耦,使得第三方服务模块的装配控制的逻辑与调用者的业务代码分离,而不是耦合在一块儿。应用程序能够根据实际业务状况启用框架扩展或替换框架组件。
缺点:
来源:https://my.oschina.net/1Gk2fdm43/blog/4336812 欢迎关注公众号 【码农开花】一块儿学习成长 我会一直分享Java干货,也会分享免费的学习资料课程和面试宝典 回复:【计算机】【设计模式】【面试】有惊喜哦