Java-SPI机制详解

打印 上一主题 下一主题

主题 771|帖子 771|积分 2313

Java之SPI机制详解

1: SPI机制简介

SPI 全称是 Service Provider Interface,是一种 JDK 内置的动态加载实现扩展点的机制,通过 SPI 技术我们可以动态获取接口的实现类,不用自己来创建。这个不是什么特别的技术,只是 一种设计理念。
2: SPI原理


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

调用者根据实际使用需要 启用、扩展、或者替换框架的实现策略
下面是一些使用了该机制的场景

  • JDBC驱动,加载不同数据库的驱动类
  • Spring中大量使用了SPI,比如:对servlet3.0规范对ServletContainerInitializer的实现、自动类型转换Type Conversion SPI(Converter SPI、Formatter SPI)等
  • Dubbo中也大量使用SPI的方式实现框架的扩展, 不过它对Java提供的原生SPI做了封装,允许用户扩展实现Filter接口
  • Tomcat 加载 META-INF/services下找需要加载的类
  • SpringBoot项目中 使用@SpringBootApplication注解时,会开始自动配置,而启动配置则会去扫描META-INF/spring.factories下的配置类
4: 源码论证

4.1 应用程序调用ServiceLoader.load方法
ServiceLoader.load方法内先创建一个新的ServiceLoader,并实例化该类中的成员变量
  1.     private static final String PREFIX = "META-INF/services/";
  2.   private ServiceLoader(Class<S> svc, ClassLoader cl) {
  3.         service = Objects.requireNonNull(svc, "Service interface cannot be null");
  4.         loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl;
  5.         acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null;
  6.         reload();
  7.     }
  8.         /**
  9.      *
  10.      * 在调用该方法之后,迭代器方法的后续调用将延迟地从头开始查找和实例化提供程序,就像新创建的加载程序所做的                  那样
  11.      */
  12.    public void reload() {
  13.         providers.clear(); //清除此加载程序的提供程序缓存,以便重新加载所有提供程序。
  14.         lookupIterator = new LazyIterator(service, loader);
  15.     }
  16.         private class LazyIterator implements Iterator<S>{
  17.         Class<S> service;
  18.         ClassLoader loader;
  19.         Enumeration<URL> configs = null;
  20.         Iterator<String> pending = null;
  21.         String nextName = null;
  22.         private boolean hasNextService() {
  23.             if (nextName != null) {
  24.                 return true;
  25.             }
  26.             if (configs == null) {
  27.                 try {
  28.                     //找到配置文件
  29.                     String fullName = PREFIX + service.getName();
  30.                     //加载配置文件中的内容
  31.                     if (loader == null)
  32.                         configs = ClassLoader.getSystemResources(fullName);
  33.                     else
  34.                         configs = loader.getResources(fullName);
  35.                 } catch (IOException x) {
  36.                     fail(service, "Error locating configuration files", x);
  37.                 }
  38.             }
  39.             while ((pending == null) || !pending.hasNext()) {
  40.                 if (!configs.hasMoreElements()) {
  41.                     return false;
  42.                 }
  43.                 //解析配置文件
  44.                 pending = parse(service, configs.nextElement());
  45.             }
  46.             //获取配置文件中内容
  47.             nextName = pending.next();
  48.             return true;
  49.         }
  50.     }
  51.                 /**
  52.              *
  53.              *  通过反射 实例化配置文件中的具体实现类
  54.              */
  55.                 private S nextService() {
  56.             if (!hasNextService())
  57.                 throw new NoSuchElementException();
  58.             String cn = nextName;
  59.             nextName = null;
  60.             Class<?> c = null;
  61.             try {
  62.                 c = Class.forName(cn, false, loader);
  63.             } catch (ClassNotFoundException x) {
  64.                 fail(service,
  65.                      "Provider " + cn + " not found");
  66.             }
  67.             if (!service.isAssignableFrom(c)) {
  68.                 fail(service,
  69.                      "Provider " + cn  + " not a subtype");
  70.             }
  71.             try {
  72.                 S p = service.cast(c.newInstance());
  73.                 providers.put(cn, p);
  74.                 return p;
  75.             } catch (Throwable x) {
  76.                 fail(service,
  77.                      "Provider " + cn + " could not be instantiated",
  78.                      x);
  79.             }
  80.             throw new Error();          // This cannot happen
  81.         }
复制代码
5: 实战

步骤1 新建以下类
  1. public interface IService {
  2.     /**
  3.      * 获取价格
  4.      * @return
  5.      */
  6.     String getPrice();
  7.     /**
  8.      * 获取规格信息
  9.      * @return
  10.      */
  11.     String getSpecifications();
  12. }
复制代码
  1. public class GoodServiceImpl implements IService {
  2.     @Override
  3.     public String getPrice() {
  4.         return "2000.00元";
  5.     }
  6.     @Override
  7.     public String getSpecifications() {
  8.         return "200g/件";
  9.     }
  10. }
复制代码
  1. public class MedicalServiceImpl implements IService {
  2.     @Override
  3.     public String getPrice() {
  4.         return "3022.12元";
  5.     }
  6.     @Override
  7.     public String getSpecifications() {
  8.         return "30粒/盒";
  9.     }
  10. }
复制代码
步骤2、在 src/main/resources/ 下建立 /META-INF/services 目录, 新增一个以接口命名的文件 org.example.IService.txt 。内容是要应用的实现类,我这边需要放入的数据如下
  1. org.example.GoodServiceImpl
  2. org.example.MedicalServiceImpl
复制代码
步骤3、使用 ServiceLoader 来加载配置文件中指定的实现。
  1. public class Main {
  2.     public static void main(String[] args) {
  3.         final ServiceLoader<IService> serviceLoader = ServiceLoader.load(IService.class);
  4.         serviceLoader.forEach(service -> {
  5.             System.out.println(service.getPrice() + "=" + service.getSpecifications());
  6.         });
  7.     }
  8. }
复制代码
输出:
  1. 2000.00元=200g/件
  2. 3022.12元=30粒/盒
复制代码
6: 优缺点

6.1 优点

解耦  使得第三方服务模块的装配控制的逻辑与调用者的业务代码分离,而不是耦合在一起,应用程序可以根据实际业务情况启用框架扩展或替换框架组件。相比使用提供接口jar包,供第三方服务模块实现接口的方式,SPI的方式使得源框架,不必关心接口的实现类的路径
6.2 缺点


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

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

铁佛

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表